1. 线程的创建

 我们在创建一个线程的时候都是这么做的

Thread thread = new Thread(() -> {
            IntStream.range(1, 1000).forEach(item -> System.out.println(Thread.currentThread().getName() + "---->" + item));
        });

 直接使用的带有Runnable接口作为参数的构造器。直接进入这个构造器,进行查看,会看到Thread对象的任何构造方法都会调用init方法进行初始化。

private void init(ThreadGroup g, Runnable target, String name,
                      long stackSize) {
        init(g, target, name, stackSize, null);
    }

    private void init(ThreadGroup g, Runnable target, String name,
                      long stackSize, AccessControlContext acc) {
        if (name == null) {
            throw new NullPointerException("name cannot be null");
        }

        this.name = name.toCharArray();

        Thread parent = currentThread();
        SecurityManager security = System.getSecurityManager();
        ......
    }

 在初始化的过程中,会对线程的各种属性进行设置。例如线程的名称,线程所属的线程组,是否是守护线程,线程的优先级,上下文加载器等。这里讲解几个需要注意的。

属性名

说明

group

线程组,允许线程访问有关其自己的线程组的信息,但不允许访问有关其线程组的父线程组或任何其他线程组的信息。

daemon

是否作为守护线程,在主线程结束的时候,守护线程退出。默认的是父线程的属性

priority

线程的优先级.默认的是父线程的优先级

contextClassLoader

这个线程的类加载器。这个可以用来破坏双亲委派模型。

stackSize

栈调用深度,默认为0。

2.线程的几种状态

 线程的状态在Thread类中由枚举类State中表示

public enum State {
       //尚未启动的线程的线程状态
        NEW,
      //可运行线程的线程状态
        RUNNABLE,
      //线程的线程状态被阻塞等待监视器锁定
        BLOCKED,
    //等待线程的线程状态
        WAITING,
    //具有指定等待时间的等待线程的线程状态。
        TIMED_WAITING,
    //终止线程的线程状态。
        TERMINATED;
    }

 关于线程的状态之间的转换,网上很多相应的文章,这里在网上找一个





java如何获取线程id并且控制其状态 获取线程对象的方法_线程状态


线程状态

这里对这些方法进行说明:

方法

说明

start

使此线程调用该线程的run方法开始执行。调用start是创建一个线程,并调用run方法。如果只调用run,并不会创建线程,而是直接在当前线程中进行执行

yield

当前线程让出cpu,进入RUNNABLE状态,但是不会释放锁。

sleep

让当前线程进入TIMED_WAITING状态,当前线程不会退出cpu的时间片占用

join

被当前线程join的线程进入TIMED_WAITING或者WAITING状态,被join的线程会退出cpu的占用

wait

调用该方法的线程进入TIMED_WAITING或者WAITING状态,会释放锁

notify/notifyAll

使调用了wait方法进入TIMED_WAITING或者WAITING状态的线程进入RUNNABLE状态

run

运行这个线程的run方法

 方法之间的比较

sleep

wait

不会释放锁

释放锁

yield

wait

不会释放锁

释放锁

进入RUNNABLE状态

TIMED_WAITING或者WAITING状态

3.其余方法

方法

说明

activeCount

获取当前线程所在线程组及其子线程组中的活跃线程数量

holdsLock

当前线程是否持有指定的锁对象

isAlive

当前线程是否存活

getState

获取当前线程状态

4.关于线程的start跟run方法的说明

先说一下这两个方法的区别

start

run

会调用底层jvm启动一个新的线程,并回调定义的run方法

不会启动一个新线程,只是回直接执行run方法

这里对于jvm创建线程进行一些说明,首先这里需要提到一个线程标准。POSIX线程简称Pthreads,标准定义了创建和操纵线程的一整套API。在Linux跟windows系统中都有对这个模型的实现,而在jvm的底层实现中也是使用这一套API进行的。