Java两个线程使用最基础wait/notify轮流打印数字和字符

news/2025/2/21 2:37:00/

背景:

最基础的java线程协同工作题目,也是笔试常见题目。 题目要求两个线程轮流打印数字(1-26)和字符(a到z)。

代码

class PrintNumRunnable implements Runnable {final Object object;final static int MAX_LOOP = 26;public PrintNumRunnable(Object object) {this.object = object;}@Overridepublic void run() {int count = 0;while (count++ < MAX_LOOP) {synchronized (object) {try {object.wait();} catch (InterruptedException e) {e.printStackTrace();}System.out.print(count);object.notify();}}// end of while}
}class PrintCharRunnable implements Runnable {final Object object;final static char MAX_CHAR = 'z';public PrintCharRunnable(Object object) {this.object = object;}@Overridepublic void run() {for(char count = 'a'; count <= MAX_CHAR; count++) {synchronized (object) {System.out.print(count);object.notify();try {object.wait();} catch (InterruptedException e) {e.printStackTrace();}}}// end of while}
}public class TwoThreadAlternatelyPrintDemo
{public static void main(String[] args) throws InterruptedException {System.out.println("main thread start ...");Object threadSyncObj  = new Object();PrintNumRunnable printNum = new PrintNumRunnable(threadSyncObj);PrintCharRunnable printChar = new PrintCharRunnable(threadSyncObj);Thread numThread = new Thread(printNum);Thread charThread = new Thread(printChar);// 启动顺便不能变,先numThread,后charThreadnumThread.start();charThread.start();// wait to finishnumThread.join();charThread.join();System.out.println("\nmain thread end");}
}

解释

numThread先启动,进入wait阻塞状态
charThread启动后,打印一个字符,然后通知numThread,自己就wait
numThread得到通知后打印数字,通知charThread, 自己进入循环的下一个迭代,并重新wait
charThread收到通知,打印一个字符,然后通知numThread,自己就wait

依次循环最后,当字符到z的时候, charThread进入wait
当数字打印26后,发送notify,自己退出线程。 charThread收到notify后也退出线程。主线程的join就结束了

代码中特意提到了不能改变两个thread的启动顺序。这是因为charThread.start();先启动了,打印了a,然后他就notify后,立刻进入wait阻塞状态了,此时numThread线程可能还未启动,没有进入到wait,当numThread正式启动后,进入wait时已经错过了charThread之前对它发送的notify消息了,此时两个线程都进入了wait状态。 可以想想是不是可以增加其他flag避免此类状况

后记

其实可以使用多种方式实现该打印题目,比如阻塞队列,打印数字的想成打印后,需要向阻塞队列put, 打印字符的线程从阻塞队列take。 阻塞队列只能容量1个元素。 这里不在展示

BlockingDeque<String> blockingDeque = (BlockingDeque) new ArrayBlockingQueue(1);

或者使用LockSupport, 相比之下wait和notify、 阻塞队列都是比较常用,能快速想起来的方式


http://www.ppmy.cn/news/1154440.html

相关文章

【小余送书第三期】CTF/AWD竞赛标准参考书+实战指南:《AWD特训营》,参与活动,领书咯!

目录 一、背景介绍 二、内容简介 三、读者对象 四、本书目录 五、书籍概览 一、背景介绍 随着网络安全问题日益凸显&#xff0c;国家对网络安全人才的需求持续增长&#xff0c;其中&#xff0c;网络安全竞赛在国家以及企业的人才培养和选拔中扮演着至关重要的角色。 在数…

ORACLE XXX序列 goes below MINVALUE 无法实例化的处理办法

--序列增加区分 --删除未使用序列表 DECLARE V_CNT INT; BEGINSELECT COUNT(*) INTO V_CNT FROM USER_SEQUENCES WHERE SEQUENCE_NAME SEQ_INTELLECT_BIZ_DETAIL_ID;IF V_CNT1 THEN BEGINEXECUTE IMMEDIATE DROP SEQUENCE SEQ_INTELLECT_BIZ_DETAIL_ID;END;END IF; END; / ---…

vue3 v-html中使用v-viewer

安装&#xff1a;npm install v-viewernext 在main.js中配置 import “viewerjs/dist/viewer.css”; import Viewer from “v-viewer”; app.use(Viewer, { Options: { inline: true, //默认值&#xff1a;false。启用内联模式。 button: true, //在查看器的右上角显示按钮。 …

uniapp 自定义tabbar页面不刷新

最近在做自定义tabbar时&#xff0c;每次切换页面都要刷新&#xff0c;页面渲染很慢&#xff0c;需要实现切换页面不刷新问题。 结局思路&#xff0c;原生的tabbar切换页面时就不选新&#xff0c;用switchTab来跳转 1.pages.json中配置tabbar&#xff0c;如下,设置高度为0&am…

三、静态路由实验

拓扑图&#xff1a; 两个路由器分了三个网段出来&#xff0c;首先对两台PC机进行配置 进入R1路由器对两边链路进行ip配置 对AR2进行相同的配置&#xff0c;然后我们查看R1的路由表&#xff0c;里面有一些直连的信息。 三个网段的设备现在可以互通&#xff0c;我们要实现跨网段…

js深拷贝与浅拷贝

1.浅拷贝概念 浅拷贝是其属性与拷贝源对象的属性共享相同引用&#xff0c;当你更改源或副本时&#xff0c;也可能&#xff08;可能说的是只针对引用数据类型&#xff09;导致其他对象也发生更改。 特性&#xff1a; 会新创建一个对象&#xff0c;即objobj2返回fasle&#xf…

python每日一练(7)

&#x1f308;write in front&#x1f308; &#x1f9f8;大家好&#xff0c;我是Aileen&#x1f9f8;.希望你看完之后&#xff0c;能对你有所帮助&#xff0c;不足请指正&#xff01;共同学习交流. &#x1f194;本文由Aileen_0v0&#x1f9f8; 原创 CSDN首发&#x1f412; 如…

虹科 | 解决方案 | 新能源车EV测试解决方案

虹科Pico与世界主流的汽车厂商保持紧密合作&#xff0c;为新能源车EV测试提供完善的解决方案。 虹科Pico新能源车EV测试解决方案 高压系统安全检测——0V残留电压测试和绝缘电阻测试 三相电机绕组微电阻平衡和高压线路连接点微电阻检测 充电通讯与高压上/下电测试 旋变/位置…