java多线程之wait(),notify(),notifyAll()的详解分析

本篇文章是对java多线程 wait(),notify(),notifyAll()进行了详细的分析介绍,需要的朋友参考下

wait(),notify(),notifyAll()不属于Thread类,而是属于Object基础类,也就是说每个对象都有wait(),notify(),notifyAll()的功能.因为每个对象都有锁,锁是每个对象的基础,当然操作锁的方法也是最基础了。

wait导致当前的线程等待,直到其他线程调用此对象的 notify() 方法notifyAll() 方法,或被其他线程中断。wait只能由持有对像锁的线程来调用

notify唤醒在此对象监视器上等待的单个线程。如果所有线程都在此对象上等待,则会选择唤醒其中一个线程(随机)。直到当前的线程放弃此对象上的锁,才能继续执行被唤醒的线程。同Wait方法一样,notify只能由持有对像锁的线程来调用.notifyall也一样,不同的是notifyall会唤配所有在此对象锁上等待的线程。

"只能由持有对像锁的线程来调用"说明wait方法与notify方法必须在同步块内执行,即synchronized(obj)之内.再者synchronized代码块内没有锁是寸步不行的,所以线程要继续执行必须获得锁。相辅相成。

一个很经典的例子(生产者与消费者):

首先是消费者线程类:

复制代码 代码如下:import java.util.List;public class Consume implements Runnable { private List container = null; private int count; public Consume(List lst) {  this.container = lst; } public void run() {  while (true) {   synchronized (container) {    if (container.size() == 0) {     try {      container.wait();// 容器为空,放弃锁,等待生产     } catch (InterruptedException e) {      e.printstacktrace();     }    }    try {     Thread.sleep(1000);    } catch (InterruptedException e) {     e.printstacktrace();    }    container.remove(0);    container.notify();    System.out.println("我吃了" + (++count) + "个");   }  } }}接下来是生产者线程类:复制代码 代码如下:import java.util.List;public class Product implements Runnable { private List container = null; private int count; public Product(List lst) {  this.container = lst; } public void run() {  while (true) {   synchronized (container) {    if (container.size() > MultiThread.MAX) {     // 如果容器超过了最大值,就不要在生产了,等待消费     try {      container.wait();     } catch (InterruptedException e) {      e.printstacktrace();     }    }    try {     Thread.sleep(1000);    } catch (InterruptedException e) {     e.printstacktrace();    }    container.add(new Object());    container.notify();    System.out.println("我生产了" + (++count) + "个");   }  } }}最后是测试类:复制代码 代码如下:import java.util.ArrayList;import java.util.List;public class MultiThread { private List container = new ArrayList(); public final static int MAX = 5; public static void main(String args[]) {  MultiThread m = new MultiThread();  new Thread(new Consume(m.getContainer())).start();  new Thread(new Product(m.getContainer())).start(); } public List getContainer() {  return container; } public void setContainer(List container) {  this.container = container; }}运行结果如下所示:复制代码 代码如下:我生产了1个我吃了1个我生产了2个我生产了3个我生产了4个我生产了5个我生产了6个我生产了7个我吃了2个我生产了8个我吃了3个我生产了9个我吃了4个我吃了5个我吃了6个我吃了7个我吃了8个我生产了10个我生产了11个我吃了9个我生产了12个我吃了10个......

相关文章

HashMap是Java中最常用的集合类框架,也是Java语言中非常典型...
在EffectiveJava中的第 36条中建议 用 EnumSet 替代位字段,...
介绍 注解是JDK1.5版本开始引入的一个特性,用于对代码进行说...
介绍 LinkedList同时实现了List接口和Deque接口,也就是说它...
介绍 TreeSet和TreeMap在Java里有着相同的实现,前者仅仅是对...
HashMap为什么线程不安全 put的不安全 由于多线程对HashMap进...