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

背景:

最基础的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.mzph.cn/news/107206.shtml

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

相关文章

旧手机热点机改造成服务器方案

如果你也跟我一样有这种想法, 那真的太酷了!!! ok,前提是得有root,不然体验大打折扣 目录 目录 1.做一个能爬墙能走百度直连的热点机(做热点机用) 2.做emby视频服务器 3.做文件服务, 存取文件 4.装青龙面板,跑一些定时任务 5.做远程摄像头监控 6.做web服务器 7.内网穿…

51单片机点阵

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、点阵是什么&#xff1f;1.点阵的原理2. 3*3 点阵显示原理3. 8*8点阵实物图4. 8*8点阵内部原理图5. 16*16点阵实物图&#xff0c;显示原理 二、使用步骤1.先…

PostgreSQL | CTE | 使用with子句的通用表达式

CTE&#xff08;Common Table Expressions&#xff09; 简单讲&#xff0c;CTE就是日常SQL中出现的with语句&#xff0c;其原理就是通过提前将数据查询出来后作为临时结果集使用&#xff0c;可以与SELECT \ INSERT \ UPDATE \ DELETE的SQL连用。 优点 可读性强 CTE 允许你将…

老卫带你学---leetcode刷题(122. 买卖股票的最佳时机 II)

122. 买卖股票的最佳时机 II 问题 给你一个整数数组 prices &#xff0c;其中 prices[i] 表示某支股票第 i 天的价格。 在每一天&#xff0c;你可以决定是否购买和/或出售股票。你在任何时候 最多 只能持有 一股 股票。你也可以先购买&#xff0c;然后在 同一天 出售。 返回…

【vim 学习系列文章 8 -- vim中 has 函数和 let g:介绍】

文章目录 1.1 vim has 函数1.2 vim let g: 介绍 1.1 vim has 函数 在 Vim 中&#xff0c;has 是一个内置的函数&#xff0c;用于检查 Vim 是否支持某个特定的特性或者是否包含某个特定的扩展。 此函数的基本语法是&#xff1a; if has(feature)其中&#xff0c;‘feature’ …

数据结构与算法--其他算法

数据结构与算法--其他算法 1 汉诺塔问题 2 字符串的全部子序列 3 字符串的全排列 4 纸牌问题 5 逆序栈问题 6 数字和字符串转换问题 7 背包问题 8 N皇后问题 暴力递归就是尝试 1&#xff0c;把问题转化为规模缩小了的同类问题的子问题 2&#xff0c;有明确的不需要继续…

设计模式之是简单工厂模式

分类 设计模式一般分为三大类&#xff1a;创建型模式、结构型模式、行为型模式。 创建型模式&#xff1a;用于创建对象&#xff0c;共五种&#xff0c;包括单例模式、简单工厂模式、工厂方法模式、抽象工厂模式、建造者模式、原型模式。结构型模式&#xff1a;用于处理类或对…

超火爆的6 个必学持续集成工具,测试人的福音

开发人员喜欢把写的代码当成自己的孩子&#xff0c;他们会被当成艺术品一样呵护。作为家长&#xff0c;总是会认为自己的孩子是最好的&#xff0c;也会尽全力给自己的孩子最好的&#xff0c;就算有时候会超出自己的能力范围。 最终&#xff0c;孩子会走出去&#xff0c;和其他…

2023自动驾驶 车道线检测数据集

目录 2023自动驾驶 车道线检测关键数据集 下载链接 labelme标注制作数据: 可视化tusimple数据集 车道线分割项目记录-tusimple数据集处理 2023自动驾驶 车道线检测关键数据集 下载链接<

android studio检测不到真机

我的情况是&#xff1a; 以前能检测到&#xff0c;有一天我使用无线调试&#xff0c;发现调试有问题&#xff0c;想改为USB调试&#xff0c;但是半天没反应&#xff0c;我就点了手机上的撤销USB调试授权&#xff0c;然后就G了。 解决办法&#xff1a; 我这个情况比较简单&…

Redission 使用Jackson处理LocalDateTime的一些坑

(redis) 文章目录 Redission 使用Jackson处理LocalDateTime的一些坑不支持jsr310的问题&#xff08;浅坑&#xff09;准备问题解决 读取 LocalDateTime类型的值的问题&#xff08;深坑&#xff09;准备问题解决 总结 Redission 使用Jackson处理LocalDateTime的一些坑 当我们想…

TypeScript 类型兼容性

TypeScript 类型兼容性 在前端开发中&#xff0c;使用 TypeScript 可以提供更强大的类型检查和类型安全。然而&#xff0c;了解 TypeScript 中的类型兼容性是至关重要的&#xff0c;因为它涉及如何处理不同类型之间的关系&#xff0c;以及在这些类型之间进行无缝的交互。本文将…

Leetcode刷题笔记--Hot61-70

1--课程表&#xff08;207&#xff09; 主要思路&#xff1a; 用 in 记录每一门课程剩余的先修课程个数&#xff0c;当剩余先修课程个数为0时&#xff0c;将该课程加入到队列q中。 每修队列q中的课程&#xff0c;以该课程作为先修课程的所有课程&#xff0c;其剩余先修课程个数…

Kubernetes实战(三)-k8s节点设置cpu高于多少就不调度

1 k8s节点设置的概念和原理 k8s是Google开源的容器集群管理系统&#xff0c;用于自动化部署、扩展和管理容器化应用程序。在k8s中&#xff0c;Node是指容器运行的物理或虚拟机器。Node可以是一个物理机或一个虚拟机器&#xff0c;k8s通过其调度器将Pod调度到每个Node上。对于一…

人机环境系统中的“人”、“机”、“环境”

人机环境系统中的“人”、“机”、“环境”是指在一个系统中&#xff0c;人、机和环境三者的相互作用和影响&#xff0c;三者之间的相互作用和影响非常复杂&#xff0c;需要进行有效的协调和管理才能实现系统的高效运转或预期的目标。因此&#xff0c;人机环境系统的设计和优化…

vue elementui的select组件实现滑到底部分页请求后端接口

vue elementui的select组件实现滑到底部分页请求后端接口 1.实现效果2.实现原理 1.实现效果 老规矩&#xff0c;直接上最后的实现效果 2.实现原理 直接上代码 <el-form-item class"diagmosisItem" label"诊断" v-scroll"handleScroll">…

✔ ★【备战实习(面经+项目+算法)】 10.15学习时间表

✔ ★【备战实习&#xff08;面经项目算法&#xff09;】 坚持完成每天必做如何找到好工作1. 科学的学习方法&#xff08;专注&#xff01;效率&#xff01;记忆&#xff01;心流&#xff01;&#xff09;2. 每天认真完成必做项&#xff0c;踏实学习技术 认真完成每天必做&…

【微信小程序开发】基础语法篇

&#x1f973;&#x1f973;Welcome Huihuis Code World ! !&#x1f973;&#x1f973; 接下来看看由辉辉所写的关于小程序的相关操作吧 目录 &#x1f973;&#x1f973;Welcome Huihuis Code World ! !&#x1f973;&#x1f973; 一.视图层 1.数据绑定 wxml js 2 .列…

Python学习六

前言&#xff1a;相信看到这篇文章的小伙伴都或多或少有一些编程基础&#xff0c;懂得一些linux的基本命令了吧&#xff0c;本篇文章将带领大家服务器如何部署一个使用django框架开发的一个网站进行云服务器端的部署。 文章使用到的的工具 Python&#xff1a;一种编程语言&…

前端工程化知识系列(4)

目录 31. 你是否了解前端性能优化的指标&#xff0c;例如首次内容绘制&#xff08;FCP&#xff09;、最大内容绘制&#xff08;LCP&#xff09;和累计布局位移&#xff08;CLS&#xff09;&#xff1f;如何针对这些指标进行优化&#xff1f;32. 你有没有经验使用服务端渲染&…