当线程被建立并启动之后,它既不是一启动就进入了执行状态,也不是一直处于执行状态,在线程的生命周期中,它要通过新建(New)、就绪(Runnable)、运线程
行(Running)、阻塞(Blocked)和死亡(Dead)五种状态。尤为是当线程启动之后,它不能一直“霸占”着CPU独自运行,因此CPU须要在多条线程之间切换,因而线程状态也对象
会屡次在运行、阻塞之间切换。blog
当程序使用new关键字建立了一个线程以后,该线程就处于新建状态,此时它和其余Java对象同样,仅仅由Java虚拟机为其分配了内存,并初始化了其成员变量生命周期
值。此时的线程对象没有表现出任何线程的动态特征,程序也不会执行线程的线程执行体。内存
当线程对象调用了start()方法以后,该线程处于就绪状态,Java虚拟机会为其建立方法调用栈和程序计数器,处于这个状态的线程并无开始运行,它只是表示该资源
线程能够运行了。至于该线程什么时候开始运行,取决于JVM里线程调度器的调度。同步
若是处于就绪状态的线程得到了CPU,开始执行run方法的线程执行体,则该线程处于运行状态。虚拟机
当发生以下状况下,线程将会进入阻塞状态:io
线程调用sleep方法主动放弃所占用的处理器资源。变量
线程调用了一个阻塞式IO方法,在该方法返回以前,该线程被阻塞。
线程试图得到一个同步监视器,但该同步监视器正被其余线程锁持有。关于同步监视器的知识将在后面有更深刻的介绍。
线程在等待某个通知(notify)。
程序调用了线程的suspend方法将该线程挂起。不过这个方法容易致使死锁,因此程序应该尽可能避免使用该方法。
当前正在执行的线程被阻塞以后,其余线程就能够得到执行的机会了。被阻塞的线程会在合适时候从新进入就绪状态,注意是就绪状态而不是运行状态。也就是
说被阻塞线程的阻塞解除后,必须从新等待线程调度器再次调度它。
针对上面的几种状况,当发生以下特定的状况将能够解除上面的阻塞,让该线程从新进入就绪状态:
调用sleep方法的线程通过了指定时间。
线程调用的阻塞式IO方法已经返回。
线程成功地得到了试图取得同步监视器。
线程正在等待某个通知时,其余线程发出了一个通知。
处于挂起状态的线程被调用了resume恢复方法。
线程状态转换图:
线程会以如下三种方式之一结束,结束后就处于死亡状态:
run()方法执行完成,线程正常结束。
线程抛出一个未捕获的Exception或Error。
直接调用该线程的stop()方法来结束该线程——该方法容易致使死锁,一般不推荐使用。