【多线程】线程间通信及状态

news2025/1/17 0:10:10

文章目录

    • 1. 线程间的通信
      • 1.1 wait和notify
      • 1.2 notify随机唤醒
      • 1.3 notifyAll()
      • 1.4 join()
    • 2. 线程间的状态
    • 3. 验证线程的状态
      • 3.1 验证NEW、RUNNABLE、TERMINATED
      • 3.2 验证WAITING
      • 3.3 验证TIMED-WAITING
      • 3.4 验证BLOCKED
    • 4. 面试题:wait和sleep对比

1. 线程间的通信

1.1 wait和notify

由于线程之间是抢占式运行,所以无法预知线程的执行顺序,但是实际开发中我们需要协调线程间的执行先后顺序。所以我们引入了等待通知机制。
wait()方法:
会使一个线程进入堵塞,进入等待状态,等待被唤醒,不然永远不会执行,但必须搭配synchronized使用,不然就会抛出异常,执行后释放锁。
notify()方法:
会随机唤醒一个等待的线程(不是先来后到),也需要搭配synchronized使用,但是只有执行完同步代码块才会释放锁。
notifyAll()方法
一次将所有等待的线程唤醒,其他和wait方法相同。

创建线程类ThreadA.java:

public class MyThreadA extends Thread{
    private Object lock;
    public MyThreadA(Object lock){
        this.lock  = lock;
    }
    @Override
    public void run() {
        try{
            synchronized (lock){
                System.out.println(Thread.currentThread().getName() + ": begin");
                lock.wait();//死等,释放锁
                System.out.println(Thread.currentThread().getName() + ": end");
            }
        } catch (InterruptedException e){
            e.printStackTrace();
        }
    }
}

创建线程类ThreadB.java:

public class MyThreadB extends Thread{
    private Object lock;
    public MyThreadB(Object lock){
        this.lock = lock;
    }
    @Override
    public void run() {
        synchronized (lock){
            System.out.println("t2: begin");
            lock.notify();//唤醒一个线程,随机的,不释放锁
            System.out.println("t2: end");
        }
    }
}

创建运行类Running.java:

public class Running {
    public static void main(String[] args) throws InterruptedException {
        Object lock = new Object();
        MyThreadA t1 = new MyThreadA(lock);
        MyThreadB t2 = new MyThreadB(lock);
        t1.start();
        Thread.sleep(1000);
        t2.start();
    }
}

如果等待机制成立,那么运行过程应该是先启动线程t1,然后打印"Thread-0: begin",然后执行wait(),进入等待并释放锁,然后在启动t2后会先打印"t2: begin",然后唤醒t1,但是因为锁还没释放,会继续执行t2后面代码,打印"t2: end",最后释放锁,执行t1,打印"Thread-0: end",如果运行结果和我们分析相同,那么等待机制成立。
运行结果:
1.1

1.2 notify随机唤醒

修改运行类:Running.java:

public class Running {
    public static void main(String[] args) throws InterruptedException {
        Object lock = new Object();
        MyThreadA t1 = new MyThreadA(lock);
        MyThreadA t2 = new MyThreadA(lock);
        MyThreadA t3 = new MyThreadA(lock);
        MyThreadB t4 = new MyThreadB(lock);
        t1.start();
        t2.start();
        t3.start();
        Thread.sleep(1000);
        t4.start();
    }
}

多次运行结果如果不相同,则notify唤醒不遵循先来后到。
运行结果:
1.2
1.22

1.3 notifyAll()

修改线程类ThreadB.java:

public class MyThreadB extends Thread{
    private Object lock;
    public MyThreadB(Object lock){
        this.lock = lock;
    }
    @Override
    public void run() {
        synchronized (lock){
            System.out.println("t2: begin");
            lock.notifyAll();//唤醒全部线程
//            lock.notify();//唤醒一个线程,随机的,不释放锁
            System.out.println("t2: end");
        }
    }
}

运行结果:
1.3
很明显唤醒了全部线程。

1.4 join()

有时我们需要等待一个线程运行结束,在执行其他线程,这个时候我们就可以使用join()方法。

public class Test1 {
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(() -> {
            System.out.println("t1执行");
        });
        Thread t2 = new Thread(() -> {
            System.out.println("t2执行");
        });
        t2.start();
        t2.join();
        t1.start();
    }
}

分析代码,t2调用join(),那么只有t2线程执行结束才可以执行下面代码,也就是t1。那么如果是:t2执行,t1执行,那么join成立。
运行结果:
在这里插入图片描述

2. 线程间的状态

线程对象在不同时期有不同状态,这些状态都在枚举类State中,则:

public class MyThreadState {
    //线程状态,都在State枚举类中
    public static void main(String[] args) {
        for (Thread.State state :
                Thread.State.values()) {
            System.out.println(state);
        }
    }
}

运行结果:
State

NEW:已经安排了工作,但未启动的线程对象。
RUNNABLE:可以运行中的线程对象,可分为正在运行,即将运行。
BLOCKED:受阻塞,等待某个监视锁的线程对象。
WAITING:无期限的等待另一个线程执行特定操作的线程。
TIMED_WIATING:指定等待时间等待另一个线程的操作的线程对象。
TERMINATED:工作结束。

3. 验证线程的状态

3.1 验证NEW、RUNNABLE、TERMINATED

实例化一个线程未启动,这时线程就是NEW状态,当启动后,运行中就是RUNNABLE状态,等待线程运行结束,就是TERMINATED状态。

    //验证: NEW RUNNABLE TERMINATED
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(() -> {
            System.out.println("运行中: " + Thread.currentThread().getState());
        });
        System.out.println("main中: " + t1.getState());
        Thread.sleep(1000);
        t1.start();
        Thread.sleep(1000);
        System.out.println("运行结束: "  +t1.getState());
    }

运行结果:
1

3.2 验证WAITING

给一个线程上锁,不解锁,让它一直等待,则线程处于WAITING状态。

    //验证: WAITING
    public static void main(String[] args) throws InterruptedException {
        Object lock = new Object();
        Thread t1 = new Thread(() -> {
            System.out.println("运行中:");
            try{
                synchronized (lock){
                    lock.wait();
                }
            }catch (InterruptedException e){
                e.printStackTrace();
            }
        });
        t1.start();
        Thread.sleep(1000);
        System.out.println("无休止等待: " + t1.getState());
    }

运行结果:
3

3.3 验证TIMED-WAITING

给一个线程休眠,休眠中查看它的状态,则应是TIMED-WAITING状态。

    //验证: TIMED_WAITING
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = new Thread(() -> {
            System.out.println("运行中: "  + Thread.currentThread().getState());
            try {
                Thread.sleep(10000);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
        });
        t1.start();
        Thread.sleep(1000);
        System.out.println("指定时间等待: " + t1.getState());
    }

运行结果:
5

3.4 验证BLOCKED

当一个线程等待另一个线程而堵塞,则应是BLOCKED状态。

 //验证: BLOCKED
    public static void main(String[] args) throws InterruptedException {
        Object lock = new Object();
        Thread t1 = new Thread(() -> {
            System.out.println("t1: " + Thread.currentThread().getState());
            try{
                synchronized (lock){
                    Thread.sleep(10000);
                }
            }catch (InterruptedException e){
                e.printStackTrace();
            }
        });
        Thread t2 = new Thread(() -> {
           synchronized (lock){
               System.out.println("t2: " + Thread.currentThread().getState());
           }
        });
        t1.start();
        Thread.sleep(1000);
        t2.start();
        Thread.sleep(1000);
        System.out.println("t2: " + t2.getState());
    }

运行结果:
6

4. 面试题:wait和sleep对比

  1. 一个用于线程通信,一个用于线程堵塞,唯一的相同点就是使线程放弃了一部分执行时间。
  2. wait需要搭配synchronized使用,sleep不用。
  3. wait使Object的方法,sleep使Thread的静态方法。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/967743.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

人工智能轨道交通行业周刊-第58期(2023.8.28-9.3)

本期关键词:成都智慧工厂、机务段、站台地标、备案大模型、AIGC报告 1 整理涉及公众号名单 1.1 行业类 RT轨道交通人民铁道世界轨道交通资讯网铁路信号技术交流北京铁路轨道交通网上榜铁路视点ITS World轨道交通联盟VSTR铁路与城市轨道交通RailMetro轨道世界铁路…

Redis 缓存穿透击穿和雪崩

一、说明 Redis 缓存的使用,极大的提升了应用程序的性能和效率,特别是数据查询方面。但同时,它也带来了一些问题。其中,最要害的问题,就是数据的一致性问题,从严格意义上讲,这个问题无解。如果对…

C# Color颜色RGB对照表

序号Color色系颜色RGB图例1Color.AliceBlue蓝色艾丽丝蓝240,248,2552Color.AntiqueWhite白色古典白色250,235,2153Color.Aqua,Color.Cyan青色浅蓝色,蓝绿色,青色0,255,255 C# Color颜色RGB对照表_旭东怪的博客-CSDN博客 C#颜色和名称样式对照…

Nginx全家桶配置详解

源码包安装NGINX A,搭建Web Server,任意HTML页面,其8080端口提供Web访问服务,截图成功访问http(s)://[Server1]:8080并且回显Web页面。保留Server1,但是不允许直接访问Server 1,再部署1套NGINX …

8.(Python数模)马尔科夫链预测

Python实现马尔科夫链预测 马尔科夫链原理 马尔科夫链是一种进行预测的方法,常用于系统未来时刻情况只和现在有关,而与过去无关。 用下面这个例子来讲述马尔科夫链。 如何预测下一时刻计算机发生故障的概率? 当前状态只存在0(故…

1分钟实现 CLIP + Annoy + Gradio 文搜图+图搜图 系统

多模态图文搜索系统 CLIP 进行 Text 和 Image 的语义EmbeddingAnnoy 向量数据库实现树状结构索引来加速最近邻搜索Gradio 轻量级的机器学习 Web 前端搭建 文搜图 图搜图 CLIP图像语义提取功能!

生信分析Python实战练习 3 | 视频21

开源生信 Python教程 生信专用简明 Python 文字和视频教程 源码在:https://github.com/Tong-Chen/Bioinfo_course_python 目录 背景介绍 编程开篇为什么学习Python如何安装Python如何运行Python命令和脚本使用什么编辑器写Python脚本Python程序事例Python基本语法 数…

已解决下载安装Python官网安装包下载速度慢问题

本文摘要:本文已解决下载安装Python官网安装包下载速度慢的问题。 😎 作者介绍:我是程序员洲洲,一个热爱写作的非著名程序员。CSDN全栈优质领域创作者、华为云博客社区云享专家、阿里云博客社区专家博主、前后端开发、人工智能研究…

47、springboot 的 国际化消息支持--就是根据浏览器选择的语言,项目上的一些提示信息根据语言的选择进行对应的显示

springboot的国际化也是基于spring mvc 的。 springboot 的 国际化消息支持–就是根据浏览器选择的语言,项目上的一些提示信息根据语言的选择进行对应的显示。 总结下国家化自动配置: 功能实现就是: 比如一个登录页面,我们在浏览…

ORB-SLAM3复现的详细过程——配置安装及ROS和脚本运行---Ubuntu20.04

ORB-SLAM3配置安装及ROS和脚本运行---Ubuntu20.04 1. 安装所需要的依赖和包2. 修改代码及文件内容2.1 CMakeLists.txt文件的修改2.2 单目可视化代码修改2.3 环境配置文件的修改2.4 源码的修改 3.ORB-SLAM3的编译3.1 构建 ORB-SLAM3 库3.2 生成ROS节点 4.ORB-SLAM3的运行4.1 非R…

ModaHub魔搭社区专访百度智能云李莅:您认为向量数据库是一个刚需产品吗?

ModaHub魔搭社区:可以看到,大模型火了以后,向量数据库受到了特别高的关注,您是如何看待这种现象呢?您认为向量数据库是一个刚需产品吗? 李莅:是的。大模型是在今年才崭露头角,或者说…

【Python Odyssey】1-1 | Python初见面

文章目录 Python 由来Python 的历史Python 的优点Python的缺点Python 运用领域Python 环境配置Python 解释器安装确认Python的版本编写Python源代码运行程序 Python 由来 Python 是 吉多范罗苏姆(Guido van Rossum)在阿姆斯特丹为了打发无聊的圣诞节&…

三维点云转换为二维图像

文章目录 前言原理代码总结与反思实验结果展示 前言 目的:将三维点云转换为二维图像 作用: a.给点云赋予彩色信息,增强点云所表达物体或对象的辨识度; b.将三维点云中绘制的目标物体通过映射关系绘制到二维图像中,这个…

Lesson5-1:OpenCV视频操作---视频读写

学习目标 掌握读取视频文件,显示视频,保存视频文件的方法 1 从文件中读取视频并播放 在OpenCV中我们要获取一个视频,需要创建一个VideoCapture对象,指定你要读取的视频文件: 创建读取视频的对象 cap cv.VideoCapt…

Nginx安装与部署

文章目录 一,说明二,下载三,Windows下安装1,安装2,启动3,验证 四,Linux下安装1,安装2,启动3,验证 五,Nginx配置 一,说明 Nginx是一款高性能Web和反向代理服务器,提供内存少,高并发,负载均衡和反向代理服务,支持windos和linux系统 二,下载 打开浏览器,输入地址: https://ngin…

【小沐学NLP】Python使用NLTK库的入门教程

文章目录 1、简介2、安装2.1 安装nltk库2.2 安装nltk语料库 3、测试3.1 分句分词3.2 停用词过滤3.3 词干提取3.4 词形/词干还原3.5 同义词与反义词3.6 语义相关性3.7 词性标注3.8 命名实体识别3.9 Text对象3.10 文本分类3.11 其他分类器3.12 数据清洗 结语 1、简介 NLTK - 自然…

递归算法学习——目标和,组合总和,字母大小写全排列

目录 一,目标和 1.题意 2.例子 3.题目接口 4.解题思路及代码 二,组合总和 1.题意 2.例子 3.题目接口 4.解题思路及代码 三,字母大小写全排列 1.题意 2.例子 3.题目接口 4.解题思路及代码 一,目标和 1.题意 给你一个…

jsp 新能源汽车论坛网Myeclipse开发mysql数据库web结构java编程计算机网页项目

一、源码特点 JSP 新能源汽车论坛网是一套完善的java web信息管理系统,对理解JSP java编程开发语言有帮助,系统具有完整的源代码和数据库,系统主要采用B/S模式开发。开发环境为 TOMCAT7.0,Myeclipse8.5开发,数据库为Mysql5.0…

Android逆向学习(二)vscode进行双开与图标修改

Android逆向学习(二)vscode进行双开与图标修改 写在前面 这其实应该还是吾爱的第一个作业,但是写完上一个博客的时候已经比较晚了,如果继续敲机械键盘吵到室友,我怕我看不到明天的太阳,所以我决定分成两篇…

英国一大学宣布:严禁使用AI生成个人陈述

8月29日,伦敦都会大学(London Met Uni)在发给合作伙伴的邮件中表示:“我们知道人工智能技术已经被用来生成个人陈述(personal statements,即文书)。请注意,本大学不接受任何由人工智…