最近在多线程编程中用到了wait(),随之即面临notify与notifyAll的选择、本人代码中使用两者皆可,查看两者的定义后、发现两者的选择还是很讲究的。
所有处于wait状态的线程,需要使用notify或者notifyAll才能被唤醒、重新被唤醒进入锁的争夺队列。notify即随机唤醒等待中的一条线程;notifyAll即唤醒全部处于等待中的线程。
两者的使用上都存在缺陷,使用notify时、容易导致死锁,而使用notifyAll时容易导致资源竞争使用的异常。所以两者的使用得根据实际情况选择、并没有说普遍的说法。下面分别提供两者使用出错的例子供参考:
使用notify出现死锁:
public class NotifyTest { public static void main(String[] args) { final OutTurn ot = new OutTurn(); for(int i=0; i<30; i++) { new Thread(new Runnable() { public void run() { ot.isSub(); } }).start(); } try { Thread.sleep(2000); } catch (InterruptedException e) { e.printStackTrace(); } for(int i=0; i<30; i++) { new Thread(new Runnable() { public void run() { ot.isNotSub(); } }).start(); } System.out.println("Thread cread complate..."); } }
public class OutTurn { private boolean isSub = true; private int count = 0; private int threadCount = 0; public synchronized void isSub() { System.out.println(threadCount++); try { while(!isSub) { System.out.println(Thread.currentThread().getName() + "- sub --- wait --" + count); this.wait(); } System.out.println(Thread.currentThread().getName() + "- sub --- " + count); isSub = false; this.notify();\\采用notifyAll即不会死锁 } catch (Exception e) { e.printStackTrace(); } count++; } public synchronized void isNotSub() { System.out.println(threadCount++); try { while(isSub) { System.out.println(Thread.currentThread().getName() + "- Not --- wait ---" + count); this.wait(); } System.out.println(Thread.currentThread().getName() + "- Not --- " + count); isSub = true; this.notify(); } catch (Exception e) { e.printStackTrace(); } count++; } }
使用notifyAll出现异常:
package notify; import java.util.ArrayList; import java.util.List; public class IntegerMaker implements Runnable { List<Integer> container = new ArrayList<Integer>(); Integer i = 1; @Override public void run() { try { while(true) { Thread.sleep(5000); Integer a = i++; synchronized(container) { container.add(a); container.notify();//notifyAll将出现异常 } } } catch (InterruptedException e) { e.printStackTrace(); } } public Integer waitFor() { synchronized(container) { System.out.println("size"); try { Thread.sleep(1000); } catch (InterruptedException e1) { e1.printStackTrace(); } if(container.size() == 0) { try { container.wait(); } catch (InterruptedException e) { e.printStackTrace(); } } System.out.println("size: " + container.size()); return container.remove(0); } } }
package notify; public class IntegerUser implements Runnable { private IntegerMaker maker; private String name; public IntegerUser(IntegerMaker maker, String name) { this.maker = maker; this.name = name; } public void run() { Integer a = maker.waitFor(); System.out.println(this.name + ":get:" + a); } public static void main(String[] args) { IntegerMaker temp = new IntegerMaker(); Thread maker = new Thread(temp); maker.start(); new Thread(new IntegerUser(temp, "xiaoge1"), "xiaoge1").start(); new Thread(new IntegerUser(temp, "xiaoge2"), "xiaoge2").start(); new Thread(new IntegerUser(temp, "xiaoge3"), "xiaoge3").start(); } }
因此两者的选择必须根据实际情况做具体的分析、然后决定应该唤醒一个还是全部。
相关推荐
3. **notifyAll()**:与`notify()`类似,但不同之处在于它会唤醒所有在当前对象监视器上等待的线程。唤醒的线程同样需要在当前线程释放锁后才能继续执行。 需要注意的是,`wait()`, `notify()`, 和 `notifyAll()` ...
Java中的`notify`和`notifyAll`是多线程编程中的关键方法,它们用于线程间的通信,特别是在处理同步机制时。这两个方法都是在`Object`类中定义的,因此可以应用于任何Java对象。 首先,我们需要理解Java的同步机制...
wait, notify 和 notifyAll,这些在多线程中被经常用到的保留关键字,在实际开发的时候很多时候却并没有被大家重视。本文对这些关键字的使用进行了描述。 在 Java 中可以用 wait、notify 和 notifyAll 来实现...
本文将深入探讨`notify()`和`notifyAll()`的区别及其在实际使用中的注意事项。 `wait()`方法使当前线程进入等待状态,并释放它所持有的对象锁。只有在持有该对象锁的情况下,线程才能调用`wait()`。当其他线程调用...
6. notify 和 notifyAll的区别notify方法只唤醒一个等待(对象的)线程并使该线程开始执行,而notifyAll 会唤醒所有等待(对象的)线程。 五、wait、notify、notifyAll方法的使用场景 wait、notify、notifyAll方法...
1.3 notify、notifyAll的区别2 两个比较经典的使用案例2.1 案例1 — ABCABC。。。三个线程顺序打印问题2.1.1 题目2.1.2 题目分析2.1.3 我的答案2.2 生产者消费者问题2.2.1 题目2.2.2 题目分析2.2.3 我的答案 源码...
源码—Java多线程5—死锁和wait notify notifyAll
总结来说,`notify`和`notifyAll`的主要区别在于唤醒线程的数量和选择策略。`notify`只唤醒一个线程,但不保证其是合适的线程;而`notifyAll`唤醒所有等待的线程,增加竞争,通常用于确保至少有一个线程可以继续执行...
Java多线程基础 线程的等待与唤醒(wait、notify、notifyAll) Java多线程基础中的线程等待与唤醒是一种常见的线程协作机制,通过wait()、notify()和notifyAll()方法来实现线程之间的同步和通信。在本节中,我们将...
本文将深入探讨Java中的wait()、notify()和notifyAll()方法,以及synchronized关键字和原子操作在多线程环境中的应用。 1. **wait()方法**: - wait()是Object类的一个方法,它的作用是让当前线程暂停执行并释放它...
Java 中Object的wait() notify() notifyAll()方法使用 在Java并发编程中,Object的wait()、notify()和notifyAll()方法是非常重要的概念,这三个方法都是Object类的方法,可以认为任意一个Object都是一种资源(或者...
wait()、notify()和notifyAll()方法2---马克-to-win java视频
wait()、notify()和notifyAll()方法1---马克-to-win java视频
`wait()`, `notify()`和`notifyAll()`与`sleep()`的主要区别在于: - **权限和锁**:`wait()`, `notify()`和`notifyAll()`需要在同步上下文中使用,而`sleep()`则不需要。 - **资源释放**:`wait()`会释放对象锁,...
`Condition`接口提供了类似`wait()`, `notify()`, `notifyAll()`的功能,但与`Lock`对象关联,提供了更好的可读性和可控性。 总之,`wait()`, `notify()`, 和 `notifyAll()` 在Java多线程编程中扮演着核心角色,...
wait方法是Object类的一个方法,用于让当前线程进入等待状态,直到其他线程调用notify或notifyAll方法来唤醒它。在wait方法中,当前线程会释放它所占有的锁,并进入等待状态。wait方法可以带有参数,指定等待的时间...
与notify()类似,notifyAll()会唤醒所有等待该对象锁的线程。然而,这些线程仍然需要在获得锁后才能继续执行,这意味着只有一个线程能够在其他线程之前获取到锁并继续运行。 4. **Lock接口**: 为了解决`...
当其他线程调用同一个对象的 `notify()` 或 `notifyAll()` 方法时,等待线程有机会重新获得锁并继续执行。需要注意的是,调用 `wait()` 方法必须在 `synchronized` 块或方法中进行,否则会抛出 `...
本文总结了Java多线程与并发系列的22道高频面试题,涵盖了多线程的实现方法、线程停止、notify和notifyAll的区别、sleep和wait的区别、volatile关键字的作用等多方面的知识点。 一、Java中实现多线程的方法 Java中...
与notify()方法不同的是,notifyAll()方法会唤醒所有等待的线程,而不是选择性的唤醒一个线程。 通过wait(),notify(),notifyAll()方法,我们可以实现线程之间的通信和同步。例如,在生产者消费者模式中,生产者线程...