[Java]线程生命周期与线程通信

这篇具有很好参考价值的文章主要介绍了[Java]线程生命周期与线程通信。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【版权声明】未经博主同意,谢绝转载!(请尊重原创,博主保留追究权)
https://www.cnblogs.com/cnb-yuchen/p/18162522
出自【进步*于辰的博客】

线程生命周期与进程有诸多相似,所以我们很容易将两者关联理解并混淆,一些细节之处确有许多不同,因为线程调度与进程调度虽都由CPU完成,但两者并不相同。
特意耗费一些时间,系统地对线程生命周期与线程通信进行梳理、整理。

目录
  • 1、线程生命周期
    • 1.1 JDK1.8版本
    • 1.2 早期版本(JDK1.2之前)
    • 1.3 落到实处
  • 2、线程通信
    • 2.1 使用 volatile 关键字
    • 2.2 使用Object类的wait()notify()
    • 2.3 使用JUC工具类 CountDownLatch
    • 2.4 使用 ReentrantLock 结合 Condition
    • 2.5 使用 LockSupport
  • 最后

1、线程生命周期

1.1 JDK1.8版本

启发博文:《线程的生命周期及五种基本状态》(转发)。

引用其中一张线程生命周期图:

[Java]线程生命周期与线程通信

在启发博文中,博主对线程五大状态和生命周期进行了很详细的说明,大家可以先行查阅。

这张图对线程生命周期总结得比较全面,我一一梳理、核对后觉得稍有不妥之处,略作修改后作如下图:

[Java]线程生命周期与线程通信

在此我先简述一下我对线程五个状态的理解:

  1. new(新建):在线程创建后、启动(start())之前所处的状态。
  2. Runnable(就绪):线程新建后并不是直接开始运行,而是被加入到等待队列,等待线程调度(等待CPU),此时就处于就绪状态。因此,这两种情况将进入就绪状态:(1)调用start();(2)因某种原因(如:线程通信、等待IO)进入阻塞状态后重新等待运行。
  3. Running(运行):线程正在运行时的状态。
  4. Blocked(阻塞):线程因某种原因(如:线程通信、等待IO)而停止运行后进入的状态。
  5. Dead(死亡):线程正常结束或异常终止后所处的状态。

相信大家在阅读完以上简述后,对线程的五大状态已经有了一个初步的认识,那状态间是如何转换的?又怎么理解呢?对于这两个问题,由于涉及到各个方法的业务和底层逻辑,本篇文章不便一一详述。如果大家想要进一步了解,可移步 → 《Thread类源码解析》。

其中,Blocked状态可能不太好理解,那位博主将其划分为三种情况:等待阻塞、同步阻塞和其他阻塞。我赞同,大家可移步启发博文查阅详述,在此不赘述,仅稍作说明:

三种阻塞情况的变动主要因“线程通信”引起,变化仅是阻塞情况的变化,状态不变,仍是Blocked

点出两个问题:
1:为什么调用notify()/notifyAll(),线程由等待Blocked变为锁定Blocked
文章排版考虑,在下文【使用Object类的wait()notify()】中说明。

2:interrupt()可中断线程,那么可中断正在阻塞的线程吗?
本质上说,可以,但会抛出异常(即不可以,故我未将其写入上图),在上文我给出的《Thread类源码解析》文章中有具体说明。

1.2 早期版本(JDK1.2之前)

相信能坚持阅读到这的博友,大部分是站在Java门槛上或刚入门不久的Java小白,你们现在了解和学习线程生命周期,获得的是已更新、迭代后的知识。个人认为,大家不需要掌握已过时的知识,但不能不了解,我先抛出两个问题:

  1. “挂起”状态是什么?怎么不在线程五大状态之列?
  2. 相信大家在一些资料中,可能见到过suspend()、resume()、stop()destroy()这4个方法,怎么上图中没有?为什么不用了?

当然是有的,只是过时了,所以没放上去,完整的图是这样:

[Java]线程生命周期与线程通信

OK,现在回答那两个问题。

“挂起”状态是一种类似Runnable(就绪)状态的状态,不同之处是进入就绪状态的线程,会释放所持有的“同步锁”,而“挂起”状态不会,“挂起”状态相当于“暂停”,故容易导致“死锁”。

为什么那4个方法会被放弃?
我寻得一答案,阐述得很详细,我便不班门弄斧了,看这里 → 《《Java面向对象编程》导读-Thread类的被废弃的suspend()、resume()和stop()方法》(转发)。

我补充一张图:
[Java]线程生命周期与线程通信

1.3 落到实处

所谓“落到实处”,就是要想掌握线程生命周期,光如上文夸夸其谈当然还不够,我们要把线程五大状态和状态间转换对应到Thread源码中才行。

如下图:

[Java]线程生命周期与线程通信

我自己感觉有点乱,源码所示如此。

当然,这不是完整图,图中状态间转换仅做了部分举例。在此,我不作说明,相信用心看到这里的博友可以大致理解。当然,也不便做出说明,因为我目前对一些方法的了解停留在“会用”的程度(见下文),并未对相应源码进行解析。

补充一点:
大家对比这张“状态图”和上文线程生命周期图,大家会发现有点对不上。

其实,WAITING就是Runnable(就绪),在线程生命周期中,一般不说“就绪”,“就绪”是进程生命周期中的术语,上文这般使用是为了方便大家理解;而RUNNABLE就是Running

2、线程通信

启发博文:《线程间通信的几种实现方式》(转发)。

我暂未整理“线程通信”相关理论,故下文将以示例的形式进行阐述。

注:以下5个示例都成功实现线程通信,输出结果是:

唤醒t1
t1已唤醒

2.1 使用 volatile 关键字

示例:

private static volatile boolean isWait = true;

public static void main(String[] args) {
    Thread t1 = new Thread(() -> {
        while (true)
            if (!isWait) {
                System.out.println("t1已唤醒");
                break;
            }
    });
    Thread t2 = new Thread(() -> {
        System.out.println("唤醒t1");
        isWait = false;
    });
    t1.start();
    t2.start();
}

如果大家不了解volatile关键字,看这里。

这里线程通信利用的是volatile关键字“保证可见性”的原理。

2.2 使用Object类的wait()和notify()

示例:

Object lock = new Object();
Thread t1 = new Thread(() -> {
    synchronized (lock) {
        try {
            lock.wait();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("t1已唤醒");
    }
});
Thread t2 = new Thread(() -> {
    synchronized (lock) {
        System.out.println("唤醒t1");
//      lock.notify();// 唤醒等待队列中的一个线程,不一定是 t1
        lock.notifyAll();
    }
});
t1.start();
t2.start();

大家还记得我在【1.1】中点出的这个问题吗?

为什么调用notify()/notifyAll(),线程由等待Blocked变为锁定Blocked

答案就在以上代码的执行过程中,我给大家捋一捋。

1、t1、t2都执行,t1在t2之前启动,先获得同步锁,t2阻塞。
2、t1调用wait()进入等待状态,释放同步锁,同步锁由t2获得,t2开始运行。
3、t2调用notify()唤醒t1,但此时同步锁仍由t2持有,t1继续等待。
4、t2运行完,释放同步锁,由t1获得,t1开始运行。

OK,就是第3点。

为什么一定要同步锁?
因为wait()notify()的底层逻辑要求必须是“先等待,再唤醒”,同步锁可以保证流程的正常执行。难道真的不能去掉同步锁?例如这样:

Object lock = new Object();
Thread t1 = new Thread(() -> {
    try {
        lock.wait();
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
    System.out.println("t1已唤醒");
});
Thread t2 = new Thread(() -> {
    System.out.println("唤醒t1");
    lock.notifyAll();
});
t1.start();
t1.join();
t2.start();

很明显,不行。这样就出现了“死锁”。

t1 等待被唤醒,主线程等待 t1 运行完。

因此,必须使用同步锁,且必须是同一把锁(lock)、

2.3 使用JUC工具类 CountDownLatch

示例:

CountDownLatch latch = new CountDownLatch(1);// 这个 1 是同步状态,类似synchronized中的 count
Thread t1 = new Thread(() -> {
    try {
        latch.await();
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
    System.out.println("t1已唤醒");
});
Thread t2 = new Thread(() -> {
    System.out.println("唤醒t1");
    latch.countDown();
});
t1.start();
t2.start();

可见,无需同步锁。为何?这就要涉及CountDownLatch类的源码了。当然,我们暂且不用深入了解,理解其底层逻辑即可。

看这里 → 《这一次,彻底搞懂Java中的synchronized关键字》(转发)。

大家找到【1.同步代码块】这一栏,底层逻辑相似。

2.4 使用 ReentrantLock 结合 Condition

示例:

Lock lock = new ReentrantLock();
Condition cond = lock.newCondition();

Thread t1 = new Thread(() -> {
    lock.lock();
    try {
        cond.await();
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
    System.out.println("t1已唤醒");
    lock.unlock();
});
Thread t2 = new Thread(() -> {
    lock.lock();
    System.out.println("唤醒t1");
    cond.signal();
    lock.unlock();
});
t1.start();
t2.start();

这两条代码合起来相当于同步锁:

lock.lock();
...
lock.unlock();

2.5 使用 LockSupport

示例:

Thread t1 = new Thread(() -> {
    LockSupport.park();
    System.out.println("t1已唤醒");
});
Thread t2 = new Thread(() -> {
    System.out.println("唤醒t1");
    LockSupport.unpark(t1);
});
t1.start();
t2.start();

可见,LockSupport类不关注是否“在等待”。

最后

本文中的例子是为了方便大家理解和阐述知识点而简单举出的,旨在“阐明知识点”,简单为主,并不一定有实用性。

如果大家想要快速地掌握这些知识点,我的建议是“自测中理解”。

本文完结。文章来源地址https://www.toymoban.com/news/detail-860284.html

到了这里,关于[Java]线程生命周期与线程通信的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【C语言趣味教程】(4) 变量:代码注释 | 变量的声明 | 初始化与赋值 | 作用域与生命周期 | 局部变量与全局变量

      🔗 《C语言趣味教程》👈 猛戳订阅!!! 0x00 引入:注释的作用 \\\"程序员最讨厌两种人:一种是不写注释的人,一种是让我写注释的人。\\\" 相信大家对注释早已有所耳闻,对于注释,C 语言有两种注释风格,我们下面会逐个讲解。   但在这之前,我们先来了解了解注释的作

    2024年02月15日
    浏览(49)
  • 【React】组件生命周期、组件通信、setState

    ◼ 组件化思想的应用: ​  有了组件化的思想,我们在之后的开发中就要充分的利用它。 ​  尽可能的将页面拆分成一个个小的、可复用的组件。 ​  这样让我们的代码更加方便组织和管理,并且扩展性也更强。 ◼ React的组件相对于Vue更加的灵活和多样,按照不同的

    2024年01月20日
    浏览(43)
  • 高级进阶多线程——多任务处理、线程状态(生命周期)、三种创建多线程的方式

    Java中的多线程是一个同时执行多个线程的进程。线程是一个轻量级的子进程,是最小的处理单元。多进程和多线程都用于实现多任务处理。 但是,一般使用多线程而不是多进程,这是因为线程使用共享内存区域。它们不分配单独的内存区域以节省内存,并且线程之间的上下

    2024年02月13日
    浏览(40)
  • 【微信小程序】生命周期,插槽和组件间通信

    1.1 组件全部的生命周期函数 小程序组件可用的全部生命周期如下表所示 生命周期函数 参数 描述说明 created 无 在组件实例刚刚被创建时执行 attached 无 在组件实例进入页面节点树时执行 ready 无 在组件在视图层布局完成后执行 moved 无 在组件实例被移动到节点树另一个位置时

    2024年02月11日
    浏览(56)
  • unity的C#学习——多线程编程(线程的生命周期、创建与管理)与线程相关类

    多线程编程是 C# 一个比较难且涵盖面比较广的知识点,本文整理仓促而且属于笔记向博客,有些地方必然还存在不严谨和错误,本人会在日后的使用过程中不断完善。如果发现问题或有改进意见可以在评论区提出,我会及时修改。 线程是程序的执行流程,也被称为 轻量级进

    2024年02月12日
    浏览(45)
  • 【微信小程序】父子组件的创建、通信与事件触发;组件生命周期

    关于微信小程序中父子组件的创建、传值,以及涉及到的组件生命周期。 组件的使用可以 提高开发效率 并 确保功能在各个页面上的应用和修改的一致性 。 例如,对于一些重复的功能,比如顶部导航栏或评论区,将其提炼成组件后,我们只需要在不同的页面中引用该组件,

    2024年02月03日
    浏览(60)
  • vue04---计算属性/监听(侦听)属性/Vue生命周期/组件介绍和使用/组件间通信/ref属性

    # 1 计算属性是基于它们的依赖变量进行缓存的 # 2 计算属性只有在它的相关依赖变量发生改变时才会重新求值,否则不会变(函数只要页面变化,就会重新运算) # 3 计算属性就像Python中的property,可以把方法/函数伪装成属性 # 4 计算属性,必须有返回值 基本使用 重写过滤案

    2024年02月08日
    浏览(63)
  • 011:Mapbox GL两种方式隐藏logo和版权,个性化版权的声明

    第011个 点击查看专栏目录 本示例的目的是介绍演示如何在vue+mapbox中用两种方式隐藏logo和版权,并个性化版权的声明 。 直接复制下面的 vue+mapbox源代码,操作2分钟即可运行实现效果 示例效果

    2023年04月17日
    浏览(54)
  • 【Java基础】Java对象的生命周期

    一个类通过编译器将一个Java文件编译为Class字节码文件,然后通过JVM中的解释器编译成不同操作系统的机器码。虽然操作系统不同,但是基于解释器的虚拟机是相同的。java类的生命周期就是指一个class文件加载到类文件注销整个过程。 一个java类的完整的生命周期会经历加载

    2024年02月12日
    浏览(40)
  • JVM包含哪几部分?JVM内存模型?线程的生命周期? 对Spring AOP的理解?布隆过滤器

    JVM由三部分组成:类加载子系统、执行引擎、运行时数据区。 类加载子系统:可以根据指定的全限定名来载入类或接口。 执行引擎:负责执行那些包含在被载入类的方法中的指令。 运行时数据区:当程序运行时,JVM需要内存来存储许多内容,例如:字节码、对象、参数、返

    2024年02月16日
    浏览(47)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包