线程的生命周期_死磕 java线程系列之线程的生命周期

b79d570187b0ee05045d6d45ab7619f2.png

(手机横屏看源码更方便)

注:java源码分析部分如无特殊说明均基于 java8 版本。

简介

大家都知道线程是有生命周期,但是彤哥可以认真负责地告诉你网上几乎没有一篇文章讲得是完全正确的。

常见的错误有:就绪状态、运行中状态(RUNNING)、死亡状态、中断状态、只有阻塞没有等待状态、流程图乱画等,最常见的错误就是说线程只有5种状态。

今天这篇文章会彻底讲清楚线程的生命周期,并分析synchronized锁、基于AQS的锁中线程状态变化的逻辑。

所以,对synchronized锁和AQS原理(源码)不了解的同学,请翻一下彤哥之前的文章先熟悉这两部分的内容,否则肯定记不住这里讲的线程生命周期。

问题

(1)线程的状态有哪些?

(2)synchronized锁各阶段线程处于什么状态?

(3)重入锁、条件锁各阶段线程处于什么状态?

先上源码

关于线程的生命周期,我们可以看一下java.lang.Thread.State这个类,它是线程的内部枚举类,定义了线程的各种状态,并且注释也很清晰。

public enum State { /** * 新建状态,线程还未开始 */ NEW, /** * 可运行状态,正在运行或者在等待系统资源,比如CPU */ RUNNABLE, /** * 阻塞状态,在等待一个监视器锁(也就是我们常说的synchronized) * 或者在调用了Object.wait()方法且被notify()之后也会进入BLOCKED状态 */ BLOCKED, /** * 等待状态,在调用了以下方法后进入此状态 * 1. Object.wait()无超时的方法后且未被notify()前,如果被notify()了会进入BLOCKED状态 * 2. Thread.join()无超时的方法后 * 3. LockSupport.park()无超时的方法后 */ WAITING, /** * 超时等待状态,在调用了以下方法后会进入超时等待状态 * 1. Thread.sleep()方法后【本文由公从号“彤哥读源码”原创】 * 2. Object.wait(timeout)方法后且未到超时时间前,如果达到超时了或被notify()了会进入BLOCKED状态 * 3. Thread.join(timeout)方法后 * 4. LockSupport.parkNanos(nanos)方法后 * 5. LockSupport.parkUntil(deadline)方法后 */ TIMED_WAITING, /** * 终止状态,线程已经执行完毕 */ TERMINATED;}

流程图

线程生命周期中各状态的注释完毕了,下面我们再来看看各状态之间的流转:

755a16bbb36510fc315522e86c946f4b.png

怎么样?是不是很复杂?彤哥几乎把网上的资料都查了一遍,没有一篇文章把这个流程图完整画出来的,下面彤哥就来一一解释:

(1)为了方便讲解,我们把锁分成两大类,一类是synchronized锁,一类是基于AQS的锁(我们拿重入锁举例),也就是内部使用了LockSupport.park()/parkNanos()/parkUntil()几个方法的锁;

(2)不管是synchronized锁还是基于AQS的锁,内部都是分成两个队列,一个是同步队列(AQS的队列),一个是等待队列(Condition的队列);

(3)对于内部调用了object.wait()/wait(timeout)或者condition.await()/await(timeout)方法,线程都是先进入等待队列,被notify()/signal()或者超时后,才会进入同步队列;

(4)明确声明,BLOCKED状态只有线程处于synchronized的同步队列的时候才会有这个状态,其它任何情况都跟这个状态无关;

(5)对于synchronized,线程执行synchronized的时候,如果立即获得了锁(没有进入同步队列),线程处于RUNNABLE状态;

(6)对于synchronized,线程执行synchronized的时候,如果无法获得锁(直接进入同步队列),线程处于BLOCKED状态;

(5)对于synchronized内部,调用了object.wait()之后线程处于WAITING状态(进入等待队列);

(6)对于synchronized内部,调用了object.wait(timeout)之后线程处于TIMED_WAITING状态(进入等待队列);

(7)对于synchronized内部,调用了object.wait()之后且被notify()了,如果线程立即获得了锁(也就是没有进入同步队列),线程处于RUNNABLE状态;

(8)对于synchronized内部,调用了object.wait(timeout)之后且被notify()了,如果线程立即获得了锁(也就是没有进入同步队列),线程处于RUNNABLE状态;

(9)对于synchronized内部,调用了object.wait(timeout)之后且超时了,这时如果线程正好立即获得了锁(也就是没有进入同步队列),线程处于RUNNABLE状态;

(10)对于synchronized内部,调用了object.wait()之后且被notify()了,如果线程无法获得锁(也就是进入了同步队列),线程处于BLOCKED状态;

(11)对于synchronized内部,调用了object.wait(timeout)之后且被notify()了或者超时了,如果线程无法获得锁(也就是进入了同步队列),线程处于BLOCKED状态;

(12)对于重入锁,线程执行lock.lock()的时候,如果立即获得了锁(没有进入同步队列),线程处于RUNNABLE状态;

(13)对于重入锁,线程执行lock.lock()的时候,如果无法获得锁(直接进入同步队列),线程处于WAITING状态;

(14)对于重入锁内部,调用了condition.await()之后线程处于WAITING状态(进入等待队列);

(15)对于重入锁内部,调用了condition.await(timeout)之后线程处于TIMED_WAITING状态(进入等待队列);

(16)对于重入锁内部,调用了condition.await()之后且被signal()了,如果线程立即获得了锁(也就是没有进入同步队列),线程处于RUNNABLE状态;

(17)对于重入锁内部,调用了condition.await(timeout)之后且被signal()了,如果线程立即获得了锁(也就是没有进入同步队列),线程处于RUNNABLE状态;

(18)对于重入锁内部,调用了condition.await(timeout)之后且超时了,这时如果线程正好立即获得了锁(也就是没有进入同步队列),线程处于RUNNABLE状态;

(19)对于重入锁内部,调用了condition.await()之后且被signal()了,如果线程无法获得锁(也就是进入了同步队列),线程处于WAITING状态;

(20)对于重入锁内部,调用了condition.await(timeout)之后且被signal()了或者超时了,如果线程无法获得锁(也就是进入了同步队列),线程处于WAITING状态;

(21)对于重入锁,如果内部调用了condition.await()之后且被signal()之后依然无法获取锁的,其实经历了两次WAITING状态的切换,一次是在等待队列,一次是在同步队列;

(22)对于重入锁,如果内部调用了condition.await(timeout)之后且被signal()或超时了的,状态会有一个从TIMED_WAITING切换到WAITING的过程,也就是从等待队列进入到同步队列;

为了便于理解,彤哥这里每一条都分的比较细,麻烦耐心看完。

测试用例

看完上面的部分,你肯定想知道怎么去验证,下面彤哥就说说验证的方法,先给出测试用例。

public class ThreadLifeTest { public static void main(String[] args) throws IOException { Object object = new Object(); ReentrantLock lock = new ReentrantLock(); Condition condition = lock.newCondition(); new Thread(()->{ synchronized (object) { try { System.out.println("thread1 waiting"); object.wait();// object.wait(5000); System.out.println("thread1 after waiting"); } catch (InterruptedException e) { e.printStackTrace(); } } }, "Thread1").start(); new Thread(()->{ synchronized (object) { try { System.out.println("thread2 notify"); // 打开或关闭这段注释,观察Thread1的状态// object.notify();【本文由公从号“彤哥读源码”原创】 // notify之后当前线程并不会释放锁,只是被notify的线程从等待队列进入同步队列 // sleep也不会释放锁 Thread.sleep(10000000); } catch (InterruptedException e) { e.printStackTrace(); } } }, "Thread2").start(); new Thread(()->{ lock.lock(); System.out.println("thread3 waiting"); try { condition.await();// condition.await(200, (TimeUnit).SECONDS); System.out.println("thread3 after waiting"); } catch (InterruptedException e) { e.printStackTrace(); } finally { lock.unlock(); } }, "Thread3").start(); new Thread(()->{ lock.lock(); System.out.println("thread4"); // 打开或关闭这段注释,观察Thread3的状态// condition.signal();【本文由公从号“彤哥读源码”原创】 // signal之后当前线程并不会释放锁,只是被signal的线程从等待队列进入同步队列 // sleep也不会释放锁 try { Thread.sleep(1000000); } catch (InterruptedException e) { e.printStackTrace(); } finally { lock.unlock(); } }, "Thread4").start(); }}

打开或关闭上面注释部分的代码,使用IDEA的RUN模式运行代码,然后点击左边的一个摄像头按钮(jstack),查看各线程的状态。

注:不要使用DEBUG模式,DEBUG模式全都变成WAITING状态了,很神奇。

5aa9c87a49b6f5be849e859969de9a1f.png

彩蛋

其实,本来这篇是准备写线程池的生命周期的,奈何线程的生命周期写了太多,等下一篇我们再来一起学习线程池的生命周期吧。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java线程的生命周期可以分为以下几个阶段: 1. 新建(New):当创建一个Thread对象时,线程处于新建状态。此时线程还没有开始运行。 2. 就绪(Runnable):线程进入就绪状态后,表示该线程已经被创建并且可以被系统调度执行。但是,由于线程调度是由操作系统控制的,所以具体的执行时间是不确定的。 3. 运行(Running):当线程获得CPU资源后,进入运行状态。此时线程开始执行run()方法中的代码。 4. 阻塞(Blocked):线程在运行过程中,可能因为一些原因而暂时停止执行。比如,线程被调用了sleep()方法、等待某个条件满足、请求输入输出等。在这种情况下,线程会进入阻塞状态。 5. 等待(Waiting):线程在某些特定条件下会进入等待状态。比如,调用了wait()方法或者join()方法。此时,线程会释放持有的锁,并且进入等待队列,直到被唤醒。 6. 超时等待(Timed Waiting):与等待状态类似,但是在等待一定时间后会自动唤醒。比如,调用了sleep()方法或者wait()方法的带有超时参数的重载方法。 7. 终止(Terminated):线程执行完run()方法中的代码后,或者因为异常或错误而提前退出时,线程将进入终止状态。 这些是Java线程的基本生命周期,在实际应用中,线程的状态可能会在不同阶段之间切换。可以通过Thread类中的方法来控制线程的状态和状态转换。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值