JUC源码讲解:逐步解析 Thread.start() 源码

抛出问题

当 new Thread() 时,线程会进入 NEW 状态,如果我们想要使用线程,就需要调用 start() 方法,那么,在使用 star() 时发生了什么?有什么需要注意的?线程是怎么一步步被创建的?跟着我一起分析源码吧!

阅读源码

为了方便讲解,我先把源码贴出来,然后逐步讲解每一步的作用

public synchronized void start() {

    if (threadStatus != 0)
        throw new IllegalThreadStateException();

    group.add(this);

    boolean started = false;
    try {
        start0();
        started = true;
    } finally {
        try {
            if (!started) {
                group.threadStartFailed(this);
            }
        } catch (Throwable ignore) {
            
        }
    }
}

我们先看这个方法头部分:

public synchronized void start() {
    if (threadStatus != 0)
        throw new IllegalThreadStateException();
}

注意到了吧,是 sync!这是为了避免被多个线程同时启动!

这个if判断,如果线程状态是 NEW,threadStatus 才等于 0,其他状态的线程是不能被 star() 的,否则会抛出线程状态异常,if 是为了避免线程被重复 start

这里提一下,init后的线程状态是NEW

看下一段,加入 ThreadGroup

group.add(this);

最后一段值得一提:

boolean started = false;
try {
    start0();
    started = true;
} finally {
    try {
        if (!started) {
            group.threadStartFailed(this);
        }
    } catch (Throwable ignore) {

    }
}

变量 started 是线程的启动状态,注意到 try 块中的 start0() 方法了吗?这是个 native 方法,目的是启动线程,在 start0() 彻底结束之前,线程的状态都是 READY!执行成功后,只要线程拿到了CPU执行权,就是 RUNNING状态

在 finally 中,通过 started 判断 start0() 是否执行成功,如果没有成功,会调用 group.threadStartFailed(this); 我们点进去这个方法里看一下吧

void threadStartFailed(Thread t) {
    synchronized(this) {
        remove(t);	// 删除该线程
        nUnstartedThreads++; // start 失败的计数器
    }
}

总结

  • start() 时为了避免线程被多次 start 而使用了 sync 和 if 。if保证非NEW状态的线程是不能被 start 的。sync方法保证不会被多个线程同时 start

  • 在 start0() 执行完全成功之前,线程的状态是 READY,执行成功后,如果线程拿到了CPU执行权,状态会变为 RUNNING,他们都是 Thread.State.class 中的 RUNNABLE 状态

  • start 失败会将线程删除,并报错给父线程