class ThreadA extends Thread{ public int num = 0; public void run(){ synchronized (this){ //在此类对象上实现同步,this 指代当前对象 for (int i= 0;i<3;++i) this.num += i; try { Thread.sleep(1000); //如果ThreadA的三个示例线程在还没有进入等待状态之前就受到了notifyAll的信号 //那将会发生严重后果,因为调用notifyAll的线程只可以调用一次notifyAll,那将会造成等待 //的线程永远等待下去 //所以在此处让它小睡一会,让其他线程有时间进入等待状态 }catch (InterruptedException e){ e.printStackTrace(); } notifyAll();//通知所有在这个对象是等待的线程开始执行,在这里就是通知 TestNotify //主线程开始执行 } } public int getNum(){ return this.num; } } class ThreadB extends Thread{ private ThreadA threada; public ThreadB (ThreadA ta){ this.threada = ta; } public void run(){ System.out.println(Thread.currentThread().getName()+" is waiting."); synchronized (threada){ try { threada.wait(); }catch (InterruptedException e){ e.printStackTrace(); } System.out.println(Thread.currentThread().getName()+" "+this.threada.getNum()); } } } public class TestNotify extends Thread{ public static void main(String[] args) { ThreadA threada = new ThreadA(); new ThreadB(threada).start(); new ThreadB(threada).start(); new ThreadB(threada).start(); threada.start(); } }代码为什么只有一个线程被唤醒

时间: 2024-02-14 17:22:32 浏览: 23
这是因为在ThreadA的run方法中,先使用了synchronized(this)来对当前对象实现同步,然后才调用了notifyAll()方法来唤醒等待的线程。而在ThreadB的run方法中,也是使用了synchronized(threada)来对threada对象实现同步,然后调用了threada.wait()方法来等待ThreadA的唤醒信号。 由于notifyAll()方法只会唤醒在当前对象上等待的线程,而在ThreadB的run方法中,每个线程都使用了不同的ThreadB对象来调用synchronized(threada),因此它们在不同的对象上等待,也就无法同时被唤醒。 为了让所有线程都能被唤醒,需要将ThreadB的构造方法中的ThreadA对象改为同一个对象,例如可以将ThreadA对象作为参数传递给TestNotify类,然后在ThreadB的构造方法中使用该参数。这样所有线程就在同一个对象上等待,notifyAll()方法就能够同时唤醒它们了。
相关问题

package Unit4; class ThreadA extends Thread{ public int num = 0; public void run(){ synchronized (this){ //在此类对象上实现同步,this 指代当前对象 for (int i= 0;i<3;++i) this.num += i; notifyAll();//通知所有在这个对象是等待的线程开始执行,在这里就是通知 TestNotify //主线程开始执行 } } public int getNum(){ return this.num; } } public class TestNotify extends Thread{ public static void main(String[] args) { ThreadA threada = new ThreadA(); threada.start(); //threada 线程有执行资格,但是还没有开始执行 synchronized (threada){ try { threada.wait(); //主线程等待 threada 线程执行结束才开始执行 // 而且只有获得了当前 threada 对象的锁之后才能执行 wait,就是说在同步域内才可以执行wait, //执行wait 后放弃对象锁 }catch (InterruptedException e){ e.printStackTrace(); } } System.out.println(threada.getNum()); } }

这段代码是一个简单的线程同步的例子,其中包括 ThreadA 和 TestNotify 两个类。 ThreadA 继承了 Thread 类,并且实现了 run 方法。在 run 方法中,使用 synchronized(this) 实现了对当前对象的同步。该同步块中使用 for 循环累加了 num 变量的值,并在循环结束后使用 notifyAll() 方法通知所有等待在该对象上的线程继续执行。 TestNotify 类中的 main 方法中,首先实例化了一个 ThreadA 对象 threada,并且调用了它的 start 方法启动线程。接着,在 synchronized 块中调用了 threada 的 wait 方法,使得主线程等待 threada 线程执行结束后才开始执行。最后,输出 threada.getNum() 的值。 这段代码的作用是实现了对线程的同步控制,确保了线程的执行顺序。

class ThreadA extends Thread { public int num = 0; public void run() { synchronized (this) { for (int i = 0; i < 3; ++i) this.num += i; // notifyAll(); } } public int getNum() { return this.num; } } public class TestNotify extends Thread { public static void main(String[] args) { ThreadA threada = new ThreadA(); threada.start(); synchronized (threada){ try { threada.wait(); System.out.println("我还是执行了"); }catch (InterruptedException e){ e.printStackTrace(); } } System.out.println(threada.getNum()); } }为什么“我还是执行了”能被输出

在`ThreadA`线程中,我们使用了`synchronized`关键字来锁住对象,并在其中进行了运算,但是没有调用`notify()`或`notifyAll()`方法唤醒其他等待该对象锁的线程。 在主线程中,我们使用`synchronized`关键字锁住了`ThreadA`对象,并调用了`wait()`方法等待该对象的锁。当`ThreadA`线程执行完同步块中的运算后,释放了该对象锁,主线程获得了该对象锁并继续执行,输出了“我还是执行了”。 这里需要注意的是,主线程获得了`ThreadA`对象锁后,才能输出“我还是执行了”。如果`ThreadA`线程在主线程调用`wait()`方法前就已经执行完同步块并释放了锁,那么主线程就无法等待`ThreadA`线程的通知,也就无法输出“我还是执行了”。

相关推荐

[1]请编写两个线程类A和[1]定义一个类Storage,有成员变量 List<Integer> nums = new ArrayList<Integer>() ; [2]在Storage中添加方法push() 这个方法的实现:下面是伪 代码 public ??? void push(int num){ // 判断nums的长度是否大于等于10 if(nums的长度 >= 10){ 让当前线程等待2秒钟 再次调用它自己,尝试再次添加 push(num) }else{ 直接将num加入 nums 集合 } } 这个方法是否要同步,自行测试和考虑。 [3]在Storage中添加方法pop() : 方法的伪代码如下: public ???? int pop(){ // 判断nums中是否有数据 if(nums的长度 == 0){ 让当前线程等待2秒钟 再次调用它自己,尝试再次获取 return pop(); }else{ return 从集合中取出一个数据。 } } [4]准备线程类A,将Storage作为成员变量, 通过构造方法 初始化Storage。在run方法中循环调用Storage的push方 法100次。(记得输出一些信息) [5]准备线程类B,将Storage作为成员变量,通过构造方法初 始化Storage。在run方法中循环的调用Storage的pop方 法100次。(记得输出一些信息) [6]编写测试类,在main方法中启动线程A和B。 线程等待两秒钟的方法: try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); }B,线程A使用继承Thread类的方式 实现。线程B使用实现Runnable接口的方式实现。 [2]在线程A中循环20次输出“BGM!!!!” 在线程B中循环20次输 出“优美的舞姿!!!!”。 [3]在主线程中分别启动线程A和线程B。

最新推荐

recommend-type

员工考勤系统.docx

员工考勤系统.docx
recommend-type

基于STM32的调试模块的外设和时钟电路分析

基于STM32的调试模块的外设和时钟电路分析。回顾 CMSIS、LL、HAL 库
recommend-type

基于 UDP 的分布式毫米波雷达python代码.zip

1.版本:matlab2014/2019a/2021a 2.附赠案例数据可直接运行matlab程序。 3.代码特点:参数化编程、参数可方便更改、代码编程思路清晰、注释明细。 4.适用对象:计算机,电子信息工程、数学等专业的大学生课程设计、期末大作业和毕业设计。
recommend-type

pyzmq-25.1.1b2-cp36-cp36m-musllinux_1_1_x86_64.whl

Python库是一组预先编写的代码模块,旨在帮助开发者实现特定的编程任务,无需从零开始编写代码。这些库可以包括各种功能,如数学运算、文件操作、数据分析和网络编程等。Python社区提供了大量的第三方库,如NumPy、Pandas和Requests,极大地丰富了Python的应用领域,从数据科学到Web开发。Python库的丰富性是Python成为最受欢迎的编程语言之一的关键原因之一。这些库不仅为初学者提供了快速入门的途径,而且为经验丰富的开发者提供了强大的工具,以高效率、高质量地完成复杂任务。例如,Matplotlib和Seaborn库在数据可视化领域内非常受欢迎,它们提供了广泛的工具和技术,可以创建高度定制化的图表和图形,帮助数据科学家和分析师在数据探索和结果展示中更有效地传达信息。
recommend-type

grpcio-1.7.0-cp35-cp35m-macosx_10_7_intel.whl

Python库是一组预先编写的代码模块,旨在帮助开发者实现特定的编程任务,无需从零开始编写代码。这些库可以包括各种功能,如数学运算、文件操作、数据分析和网络编程等。Python社区提供了大量的第三方库,如NumPy、Pandas和Requests,极大地丰富了Python的应用领域,从数据科学到Web开发。Python库的丰富性是Python成为最受欢迎的编程语言之一的关键原因之一。这些库不仅为初学者提供了快速入门的途径,而且为经验丰富的开发者提供了强大的工具,以高效率、高质量地完成复杂任务。例如,Matplotlib和Seaborn库在数据可视化领域内非常受欢迎,它们提供了广泛的工具和技术,可以创建高度定制化的图表和图形,帮助数据科学家和分析师在数据探索和结果展示中更有效地传达信息。
recommend-type

zigbee-cluster-library-specification

最新的zigbee-cluster-library-specification说明文档。
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

MATLAB图像处理算法宝典:从理论到实战

![MATLAB图像处理算法宝典:从理论到实战](https://img-blog.csdnimg.cn/20200717112736401.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2d1emhhbzk5MDE=,size_16,color_FFFFFF,t_70) # 1. MATLAB图像处理基础理论 MATLAB图像处理是一种利用MATLAB编程语言进行图像处理的强大工具。它提供了丰富的函数和工具箱,用于图像获取、增强、分
recommend-type

matlab中1/x的非线性规划

在MATLAB中,可以使用非线性规划函数(`fmincon`)来优化一个包含1/x的非线性目标函数。下面是一个简单的例子: ```matlab % 定义目标函数 fun = @(x) 1/x; % 定义约束函数(这里没有约束) nonlcon = []; % 定义初始点 x0 = 1; % 定义优化选项 options = optimoptions('fmincon', 'Display', 'iter'); % 进行非线性规划 [x, fval] = fmincon(fun, x0, [], [], [], [], [], [], nonlcon, options); ``` 在
recommend-type

JSBSim Reference Manual

JSBSim参考手册,其中包含JSBSim简介,JSBSim配置文件xml的编写语法,编程手册以及一些应用实例等。其中有部分内容还没有写完,估计有生之年很难看到完整版了,但是内容还是很有参考价值的。