亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

java多線程之wait(),notify(),notifyAll()的詳解分析

 更新時(shí)間:2013年06月04日 15:20:19   作者:  
本篇文章是對(duì)java多線程 wait(),notify(),notifyAll()進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下
wait(),notify(),notifyAll()不屬于Thread類,而是屬于Object基礎(chǔ)類,也就是說每個(gè)對(duì)象都有wait(),notify(),notifyAll()的功能.因?yàn)槊總€(gè)對(duì)象都有鎖,鎖是每個(gè)對(duì)象的基礎(chǔ),當(dāng)然操作鎖的方法也是最基礎(chǔ)了。

wait導(dǎo)致當(dāng)前的線程等待,直到其他線程調(diào)用此對(duì)象的 notify() 方法或 notifyAll() 方法,或被其他線程中斷。wait只能由持有對(duì)像鎖的線程來調(diào)用。

notify喚醒在此對(duì)象監(jiān)視器上等待的單個(gè)線程。如果所有線程都在此對(duì)象上等待,則會(huì)選擇喚醒其中一個(gè)線程(隨機(jī))。直到當(dāng)前的線程放棄此對(duì)象上的鎖,才能繼續(xù)執(zhí)行被喚醒的線程。同Wait方法一樣,notify只能由持有對(duì)像鎖的線程來調(diào)用.notifyall也一樣,不同的是notifyall會(huì)喚配所有在此對(duì)象鎖上等待的線程。
"只能由持有對(duì)像鎖的線程來調(diào)用"說明wait方法與notify方法必須在同步塊內(nèi)執(zhí)行,即synchronized(obj)之內(nèi).再者synchronized代碼塊內(nèi)沒有鎖是寸步不行的,所以線程要繼續(xù)執(zhí)行必須獲得鎖。相輔相成。
看一個(gè)很經(jīng)典的例子(生產(chǎn)者與消費(fèi)者):
首先是消費(fèi)者線程類:
復(fù)制代碼 代碼如下:

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();// 容器為空,放棄鎖,等待生產(chǎn)
     } catch (InterruptedException e) {
      e.printStackTrace();
     }
    }
    try {
     Thread.sleep(1000);
    } catch (InterruptedException e) {
     e.printStackTrace();
    }
    container.remove(0);
    container.notify();
    System.out.println("我吃了" + (++count) + "個(gè)");
   }
  }
 }
}

接下來是生產(chǎn)者線程類:
復(fù)制代碼 代碼如下:

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) {
     // 如果容器超過了最大值,就不要在生產(chǎn)了,等待消費(fèi)
     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("我生產(chǎn)了" + (++count) + "個(gè)");
   }
  }
 }
}

最后是測試類:
復(fù)制代碼 代碼如下:

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;
 }
}

運(yùn)行結(jié)果如下所示:
復(fù)制代碼 代碼如下:

我生產(chǎn)了1個(gè)
我吃了1個(gè)
我生產(chǎn)了2個(gè)
我生產(chǎn)了3個(gè)
我生產(chǎn)了4個(gè)
我生產(chǎn)了5個(gè)
我生產(chǎn)了6個(gè)
我生產(chǎn)了7個(gè)
我吃了2個(gè)
我生產(chǎn)了8個(gè)
我吃了3個(gè)
我生產(chǎn)了9個(gè)
我吃了4個(gè)
我吃了5個(gè)
我吃了6個(gè)
我吃了7個(gè)
我吃了8個(gè)
我生產(chǎn)了10個(gè)
我生產(chǎn)了11個(gè)
我吃了9個(gè)
我生產(chǎn)了12個(gè)
我吃了10個(gè)
......

相關(guān)文章

最新評(píng)論