[Java][Thread需要了解的那些事之生命周期]
1.线程生命周期
生命周期2.按图分析
当线程通过new创建了一个Thread的时候,线程就处于New状态。
1.当这个Thread对象调用了start()方法的时候,这个线程就进入到了Runable状态,也就是拥有了被执行的资格,正在等待获得CPU资源。
2.当这个线程得到CPU资源,他就会进入到Running状态,这时候会执行Thread里的run()方法的任务。
3.当这个线程执行完run()方法的任务或者执行过程中出现异常,那么线程就会进入到Dead状态。
4.当这个线程调用了yield()的方法的时候,该线程就会进入到Runnable状态。继续等待获得CPU资源。
5.当这个线程调用了sleep()或者I/O阻塞的时候,就会进入到Blocked状态,当睡眠时间过了,I/O完成了,线程就会进入到Runnable状态。
6.当这个线程调用了join(),这个线程的父线程就会被阻塞,一般父线程的任务会在子线程的任务之前执行。但如果子线程调用了join(),那么父线程就必须等待子线程任务完成或者超时(join可以设置超时时间),父线程会进入到Runnable,等待获得CPU资源才能进行往下执行。
7.这是在多线程同步的情况下讨论的,当一个线程获得CPU资源后,也就是进入到了Running状态,在run()方法里调用了wait(),那么这个线程就会进入到等待Blocked状态。当其他线程调用了notify()或者notifyAll()的时候,这个线程就会进入到锁定Blocked状态,当其他线程释放了锁之后,该线程就可以回到Runable状态了。
3.过程解析
总的来说,线程的生命周期有五个过程。分别是new,runnable,running,blocked和dead.
(1) New
当创建Thread类的一个实例(对象)时,此线程进入New状态。例如:Thread thread=new Thread();处于New状态的线程有自己的内存空间。
(2) Runnable
当新建的线程对象调用start()方法的时候,线程处于Runnable状态,这时候的线程已经具备了运行条件,但还没有分配到CPU,处于线程就绪队列(尽管是采用队列形式,事实上,把它称为可运行池而不是可运行队列。因为cpu的调度不一定是按照先进先出的顺序来调度的),等待系统为其分配CPU。Runnable状态并不是执行状态,当系统选定一个处于Runnable状态的Thread对象后,它就会从等待执行状态进入执行状态,系统挑选的动作称之为“cpu调度”。
(3) Running
处于Runnable状态的线程一旦获得CPU,线程就进入Running状态并自动调用自己的run方法。处于Running状态的线程最为复杂,它可以变为Runnable状态、Blocked状态和Dead状态。
当对在Running状态的线程调用yield()方法,它就会让出CPU资源,再次变为就绪状态,当然,也会出现这种情况,线程调用yield()方法,这个线程进入到Runnable状态,他和其他线程都有机会被重新分配CPU资源,这里就有可能出现该线程调用yield()方法让出CPU资源后,处于Runnable状态的该线程又被分配到了CPU资源,又可以重新进入到Running状态。
当发生如下情况是,线程会从Running状态变为Blocked状态:
① 线程调用sleep()方法主动放弃所占用的系统资源
② 线程调用一个阻塞式IO方法,在该方法返回之前,该线程被阻塞
③ 线程试图获得一个同步监视器(synchronized锁),但同步监视器正被其他线程所持有
④ 线程在等待某个通知(notify)
⑤ 程序调用了线程的suspend方法将线程挂起。不过该方法容易导致死锁,所以程序应该尽量避免使用该方法。这个方法已经废弃了。
当线程的run()方法执行完,或者被强制性地终止,例如出现异常,或者调用了stop()、destroy()方法等等,就会从Running状态转变为Dead状态。stop()和destroy()方法已经废弃了,建议不要再使用了。
(4) Blocked
处于Running状态的线程在某些情况下,如执行了sleep()方法或等待I/O设备等资源,将让出CPU并暂时停止自己的运行,进入阻塞状态。在阻塞状态的线程不能进入Runnale队列。只有当引起阻塞的原因消除时,如睡眠时间已到,或等待的I/O设备空闲下来,线程便转入Runnale状态,重新到Runnale队列中排队等待,被系统选中后进入到Running状态,线程便从原来停止的位置开始继续运行。
(5) Dead
当线程的run()方法执行完,或者被强制性地终止,就认为它死去。这个线程对象也许是活的,但是,它已经不是一个单独执行的线程。线程一旦死亡,就不能复生。如果在一个死去的线程上调用start()方法,会抛出java.lang.IllegalThreadStateException异常。