写在开头

在前面的几篇博客里,我们学习了Java的多线程,包括线程的作用、创建方式、重要性等,那么今天我们就要正式踏入线程,去学习更加深层次的知识点了。

第一个需要学的就是线程的生命周期,也可以将之理解为线程的几种状态,以及互相之间的切换,这几乎是Java多线程的面试必考题,每一年都有大量的同学,因为这部分内容回答不够完美而错过高薪,今天我们结合源码,好好来聊一聊。

线程的生命周期

所谓线程的生命周期,就是从它诞生到消亡的整个过程,而不同的编程语言,对线程生命周期的封装是不同的,在Java中线程整个生命周期被分为了六种状态,我们下面就来一起学习一下。

线程的6种状态

对于Java中线程的状态划分,我们其实要从两个方面去看,一是JVM层面,这是我们程序运行的核心,另一层面是操作系统层面,这是我们JVM能够运行的核心。为了更直观的分析,build哥列了一个对比图:

在操作系统层面,对于RUNNABLE状态拆分为(READY、RUNNING),那为什么在JVM层面没有分这么细致呢?

这是因为啊,在当下时分多任务操作系统架构下,线程的驱动是通过获取CPU时间片,而每个时间片的间隔非常之短(10-20ms),这就意味着一个线程在cpu上运行一次的时间在0.01秒,随后CPU执行权就会发生切换,在如此高频的切换下,JVM就没必要去区分READY和RUNNING了。

在Java的源码中也可以看到,确实只分了6种状态:

【源码解析1】

// Thread.State 源码
public enum State {
//省略了每个枚举值上的注释
NEW,
RUNNABLE,
BLOCKED,
WAITING,
TIMED_WAITING,
TERMINATED;
}

NEW(初始化状态)

我们通过new一个Thread对象,进行了初始化工作,这时的线程还没有被启动。

【代码示例1】

public class Test {
public static void main(String[] args) {
//lambda 表达式
Thread thread = new Thread(() -> {});
System.out.println(thread.getState());
}
}
//执行结果:NEW

我们通过thread.getState()方法去获得当前线程所处在的状态,此时输出为NEW。

RUNNABLE(可运行状态)

对于这种状态的描述,我们来看一下Thread源码中如何说的:

/**
* Thread state for a runnable thread. A thread in the runnable
* state is executing in the Java virtual machine but it may
* be waiting for other resources from the operating system
* such as processor.
*/

大致意思是线程处于RUNNABLE状态下,代表它可能正处于运行状态,或者正在等待CPU资源的分配。

那么我们怎样从NEW状态变为RUNNABLE呢?答案很简单,我们只需要调用start()方法即可!

【代码示例2】

public class Test {
public static void main(String[] args) {
//lambda 表达式
Thread thread = new Thread(() -> {});
thread.start();
System.out.println(thread.getState());
}
}
//执行结果:RUNNABLE

BLOCKED(阻塞状态)

当线程线程进入 synchronized 方法/块或者调用 wait 后(被 notify)重新进入 synchronized 方法/块,但是锁被其它线程占有,这个时候线程就会进入 BLOCKED(阻塞) 状态。这时候只有等到锁被另外一个线程释放,重新获取锁后,阻塞状态解除!

WAITING(无限时等待)

当通过代码将线程转为WAITING状态后,这种状态不会自动切换为其他状态,是一种无限时状态,直到整个线程接收到了外界通知,去唤醒它,才会从WAITING转为uRUNNABLE。

调用下面这 3 个方法会使线程进入等待状态:

  1. Object.wait():使当前线程处于等待状态直到另一个线程唤醒它;
  2. Thread.join():等待线程执行完毕,底层调用的是 Object 的 wait 方法;
  3. LockSupport.park():除非获得调用许可,否则禁用当前线程进行线程调度。

TIMED_WAITING(有限时等待)

与WAITING相比,TIMED_WAITING是一种有限时的状态,可以通过设置等待时间,没有外界干扰的情况下,达到指定等待时间后,自动终止等待状态,转为RUNNABLE状态。

调用如下方法会使线程进入超时等待状态:

  1. Thread.sleep(long millis):使当前线程睡眠指定时间;
  2. Object.wait(long timeout):线程休眠指定时间,等待期间可以通过notify()/notifyAll()唤醒;
  3. Thread.join(long millis):等待当前线程最多执行 millis 毫秒,如果 millis 为 0,则会一直执行;
  4. LockSupport.parkNanos(long nanos): 除非获得调用许可,否则禁用当前线程进行线程调度指定时间;
  5. LockSupport.parkUntil(long deadline):同上,也是禁止线程进行调度指定时间;

TERMINATED(终止状态)

线程正常执行结束,或者异常终止,会转变到 TERMINATED 状态。

线程状态的切换

上面的6种状态随着程序的运行,代码(方法)的执行,上下文的切换,也伴随着状态的转变。

NEW 到 RUNNABLE 状态

这一种转变比较好理解,我们通过new,初始化一个Thread对象后,这时就是处于线程的NEW状态,此时线程是不会获取CPU时间片调度执行的,只有在调用了start()方法后,线程彻底创建完成,进入RUNNABLE状态,等待操作系统调度执行!这种状态是NEW -> RUNNABLE的单向转变

RUNNABLE 与 BLOCKED 的状态转变

synchronized 修饰的方法、代码块同一时刻只允许一个线程执行,其他线程只能等待,等待的线程会从 RUNNABLE 转变到 BLOCKED 状态,当等待的线程获得 synchronized 隐式锁时,就又会从 BLOCKED 转变到 RUNNABLE 状态。我们通过一段代码示例看一下:

【代码示例3】

public class Test {
public static void main(String[] args) {
Thread thread1 = new Thread(() -> {
testMethod();
});
Thread thread2 = new Thread(() -> {
testMethod();
}); thread1.start();
thread2.start(); System.out.println(thread1.getName()+":"+thread1.getState());
System.out.println(thread2.getName()+":"+thread2.getState());
}
// 同步方法争夺锁
private static synchronized void testMethod() {
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

输出:

Thread-0:RUNNABLE
Thread-1:BLOCKED

代码中在主线程中创建了2个线程,线程中都调用了同步方法,随后去启动线程,因为CPU的执行效率较高,还没阻塞已经完成的打印,所以大部分时间里会输出两线程均为RUNNABLE状态;

当CPU效率稍低时,就会呈现上述结果,thread1启动后进入RUNNABLE状态,并且获得了同步方法,这是thread2启动后,调用的同步方法锁已经被占用,它作为等待的线程会从 RUNNABLE 转变到 BLOCKED 状态,待到thread1同步方法执行完毕,释放synchronized锁后,thread2获得锁,从BLOCKED转为RUNNABLE状态。

RUNNABLE 与 WAITING 的状态转变

1、获得 synchronized 隐式锁的线程,调用无参数的 Object.wait() 方法,状态会从 RUNNABLE 转变到 WAITING;调用 Object.notify()、Object.notifyAll() 方法,线程可能从 WAITING 转变到 RUNNABLE 状态。

2、调用无参数的 Thread.join() 方法。join() 是一种线程同步方法,如有一线程对象 Thread t,当调用 t.join() 的时候,执行代码的线程的状态会从 RUNNABLE 转变到 WAITING,等待 thread t 执行完。当线程 t 执行完,等待它的线程会从 WAITING 状态转变到 RUNNABLE 状态。

3、调用 LockSupport.park() 方法,线程的状态会从 RUNNABLE 转变到 WAITING;调用 LockSupport.unpark(Thread thread) 可唤醒目标线程,目标线程的状态又会从 WAITING 转变为 RUNNABLE 状态。

RUNNABLE 与 TIMED_WAITING 的状态转变

这种与上面的很相似,只是在方法调用和参数上有细微差别,因为,TIMED_WAITING 和 WAITING 状态的区别,仅仅是调用的是超时参数的方法。

转变方法在上文中已经提到了,这里以sleep(time)为例,写一个测试案例:

【代码示例4】

public class Test {
public static void main(String[] args) throws InterruptedException { Thread thread1 = new Thread(() -> {
testMethod();
});
Thread thread2 = new Thread(() -> {
// testMethod();
});
thread1.start();
Thread.sleep(1000L);
thread2.start(); System.out.println(thread1.getName()+":"+thread1.getState());
System.out.println(thread2.getName()+":"+thread2.getState());
}
// 同步方法争夺锁
private static synchronized void testMethod() {
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

输出:

Thread-0:TIMED_WAITING
Thread-1:TERMINATED

这里面我们启动threa1后,让主线程休眠了1秒,这时thread1获得同步方法后,方法内部执行了休眠2秒的操作,因此它处于TIMED_WAITING状态,而thread2正常运行结束,状态处于TERMINATED(这个案例同样可以印证下面RUNNABLE到TERMINATED的转变)。

RUNNABLE 到 TERMINATED 状态

转变为TERMINATED状态,表明这个线程已经执行完毕,通常用如下几种情况:

  1. 线程执行完 run() 方法后,会自动转变到 TERMINATED 状态;
  2. 执行 run() 方法时异常抛出,也会导致线程终止;
  3. Thread类的 stop() 方法已经不建议使用。

总结

今天关于线程的6种状态就讲到这里啦,这是个重点知识点,希望大家能够铭记于心呀!

结尾彩蛋

如果本篇博客对您有一定的帮助,大家记得留言+点赞+收藏呀。原创不易,转载请联系Build哥!

如果您想与Build哥的关系更近一步,还可以关注“JavaBuild888”,在这里除了看到《Java成长计划》系列博文,还有提升工作效率的小笔记、读书心得、大厂面经、人生感悟等等,欢迎您的加入!

Java面试必考题之线程的生命周期,结合源码,透彻讲解!的更多相关文章

  1. Java多线程(五)线程的生命周期

    点我跳过黑哥的卑鄙广告行为,进入正文. Java多线程系列更新中~ 正式篇: Java多线程(一) 什么是线程 Java多线程(二)关于多线程的CPU密集型和IO密集型这件事 Java多线程(三)如何 ...

  2. drf复习(一)--原生djangoCBV请求生命周期源码分析、drf自定义配置文件、drf请求生命周期dispatch源码分析

    admin后台注册model  一.原生djangoCBV请求生命周期源码分析 原生view的源码路径(django/views/generic/base.py) 1.从urls.py中as_view ...

  3. Java 多线程(三) 线程的生命周期及优先级

    线程的生命周期 线程的生命周期:一个线程从创建到消亡的过程. 如下图,表示线程生命周期中的各个状态: 线程的生命周期可以分为四个状态: 1.创建状态: 当用new操作符创建一个新的线程对象时,该线程处 ...

  4. Java 多线程(三)—— 线程的生命周期及方法

    这篇博客介绍线程的生命周期. 线程是一个动态执行的过程,它也有从创建到死亡的过程. 线程的几种状态 在 Thread 类中,有一个枚举内部类: 上面的信息以图片表示如下: 第一张图: 第二张图:把等待 ...

  5. Java面试必问之线程池的创建使用、线程池的核心参数、线程池的底层工作原理

    一.前言 大家在面试过程中,必不可少的问题是线程池,小编也是在面试中被问啥傻了,JUC就了解的不多.加上做系统时,很少遇到,自己也是一知半解,最近看了尚硅谷阳哥的课,恍然大悟,特写此文章记录一下!如果 ...

  6. java多线程(二)线程的生命周期

    1.   线程生命周期 新建(New).就绪(Runnable).运行(Running).阻塞(Blocked)和死亡(Dead)5种状态. 1.1. 新建 l  new关键字创建了一个线程之后,该线 ...

  7. java多线程(二)-线程的生命周期及线程间通信

    一.摘要    当我们将线程创建并start时候,它不会一直占据着cpu执行,而是多个线程间会去执行着这个cpu,此时这些线程就会在多个状态之间进行着切换. 在线程的生命周期中,它会有5种状态,分别为 ...

  8. Java 多线程(五)—— 线程池基础 之 FutureTask源码解析

    FutureTask是一个支持取消行为的异步任务执行器.该类实现了Future接口的方法. 如: 取消任务执行 查询任务是否执行完成 获取任务执行结果(”get“任务必须得执行完成才能获取结果,否则会 ...

  9. 20道Java面试必考题

    系统整理了一下有关Java的面试题,包括基础篇,javaweb篇,框架篇,数据库篇,多线程篇,并发篇,算法篇等等,陆续更新中.其他方面如前端后端等等的面试题也在整理中,都会有的. 注:文末有福利!pd ...

  10. JAVA基础知识之多线程——线程的生命周期(状态)

    线程有五个状态,分别是新建(New).就绪(Runnable).运行(Running).阻塞(Blocked)和死亡(Dead). 新建和就绪 程序使用new会新建一个线程,new出的对象跟普通对象一 ...

随机推荐

  1. TienChin 渠道管理-更新渠道接口开发

    ChannelController /** * 修改渠道 */ @PreAuthorize("hasPermission('tienchin:channel:edit')") @L ...

  2. NLP文本匹配任务Text Matching [无监督训练]:SimCSE、ESimCSE、DiffCSE 项目实践

    NLP文本匹配任务Text Matching [无监督训练]:SimCSE.ESimCSE.DiffCSE 项目实践 文本匹配多用于计算两个文本之间的相似度,该示例会基于 ESimCSE 实现一个无监 ...

  3. PaddleHub实战篇{ERNIE实现文新闻本分类、ERNIE3.0 实现序列标注}【四】

     相关文章: 基础知识介绍: [一]ERNIE:飞桨开源开发套件,入门学习,看看行业顶尖持续学习语义理解框架,如何取得世界多个实战的SOTA效果?_汀.的博客-CSDN博客_ernie模型 百度飞桨: ...

  4. AIX6.1修改时区/修改时间

    环境 AIX6.1 修改时间 格式:date -n mmddHHMMYY  #mm表示月分,dd表示日期,HH表示小时,MM表示分钟,YY表示年份. 例子:date -n 1204171622  #修 ...

  5. 苹果M3 Max有两种版本:14+40?还是16+40?

    最近有关苹果M3系列处理器的消息突然多了起来,包括M3.M3 Pro.M3 Max,都将升级为台积电3nm工艺,但规格上比较保守,至少核心数量不会大幅增加. 此前说法称,M3 Max将配备14个CPU ...

  6. [Java]format string is malformed java

    format string is malformed java 最近在做代码审查,发现很多在使用 String.format 的时候遇到了IDEA报的 Format string 'xxx' is m ...

  7. NC224933 漂亮数

    题目链接 题目 题目描述 小红定义一个数满足以下条件为"漂亮数": 该数不是素数. 该数可以分解为2个素数的乘积. 4 是漂亮数,因为 4=2*2 21 是漂亮数,因为 21=3* ...

  8. NC17871 CSL分苹果

    题目链接 题目 题目描述 CSL手上有n个苹果,第i个苹果的质量是wi,现在他想把这些苹果分给他的好朋友wavator和tokitsukaze.但是CSL为了不让他们打架,根据质量决定尽量地均分成两堆 ...

  9. Python 写入文件、读取文件内容——open函数/readLines/Write/find函数用法

    1.读取.txt整个文件 ww.txt文件在程序文件所在的目录,在文件存储在其他地方,ww.txt需要添加文件路径,如:E:\book1\ww.txt:读取后希望返回的是列表类型,将read改为rea ...

  10. SATA 学习笔记——Link Layer 8b/10b 编码解析

    一.故事前传 我们上回说到Link layer的结构,link layer的作用大致可以包括以下几点: Frame flow control CRC的生成与检测 对数据与控制字符的Scrmable/D ...