欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

java線程間通訊的一些方法總結(jié)

 更新時間:2021年02月05日 10:57:20   作者:負債程序猿  
這篇文章主要介紹了java線程間通訊的一些方法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

前言

并發(fā)編程中,我們可能會遇到這樣一個場景

A、B兩個線程并行,但是我希望保證B線程在A線程執(zhí)行完了后再執(zhí)行

這個時候就需要線程間進行通訊

A執(zhí)行完了后對B說一聲,喂B,我執(zhí)行完了

來康康用Java怎么實現(xiàn)

1、基于synchronized

2、基于reentrantLock

3、基于volatile

4、基于countDownLatch

我目前就知道這四種

1、synchronized+wait() 和 notify()

wait() 和 notify()都是Object類的通訊方法,注意一點,wait和 notify必須搭配synchronized使用,并且wait()會釋放鎖,notify()不會釋放鎖

public class SynchronizedTest {

 //定義個year,用來記錄某明星的練習(xí)年數(shù)
 private static double year;

 public void run() {
  //線程A,練習(xí)唱跳rap
  Thread threadA = new Thread(() -> {
   synchronized (this) {
    for (year = 0.5; year <= 5; year += 0.5) {
     System.out.println("蔡徐雞開始練習(xí)唱跳rap:已練習(xí)" + year + "年");
     try {
      Thread.sleep(288);
     } catch (InterruptedException e) {
      e.printStackTrace();
     }
     //眾所周知,練習(xí)兩年半即可出道
     if (year == 2.5) {
      System.out.println("===========================>成功練習(xí)兩年半,出道?。。?);
      this.notify();
     }
    }
   }
  });
  //線程B,練習(xí)打籃球
  Thread threadB = new Thread(() -> {
   while (true) {
    synchronized (this) {
     try {
      this.wait();
     } catch (InterruptedException e) {
      e.printStackTrace();
     }
     System.out.println("蔡徐雞開始練習(xí)打籃球");
    }
   }
  });
  //注意,一定要先啟動B,不然會導(dǎo)致B永遠拿不到鎖
  threadB.start();
  threadA.start();
 }

 public static void main(String[] args) {
  SynchronizedTest test = new SynchronizedTest();
  test.run();
 }
}

運行結(jié)果:

蔡徐雞開始練習(xí)唱跳rap:已練習(xí)0.5年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)1.0年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)1.5年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)2.0年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)2.5年
===========================>成功練習(xí)兩年半,出道?。?!
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)3.0年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)3.5年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)4.0年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)4.5年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)5.0年
蔡徐雞開始練習(xí)打籃球

注意看運行結(jié)果,線程A在執(zhí)行notify后并沒有釋放鎖,而是執(zhí)行完當(dāng)前任務(wù)才開始執(zhí)行線程B的任務(wù)

2、基于ReentrantLock

ReentrantLock也能實現(xiàn)線程間通訊,不過有點麻煩,需要結(jié)合ReentrantLock的Condition

public class LockTest {
  //定義個year,用來記錄某明星練習(xí)打籃球的年數(shù)
  private static double year;

  public static void main(String[] args) {
   ReentrantLock lock = new ReentrantLock();
   Condition condition = lock.newCondition();
   //線程A,練習(xí)唱跳rap
   Thread threadA = new Thread(() -> {
    //執(zhí)行業(yè)務(wù)代碼前上鎖
    lock.lock();
    for (year = 0.5; year <= 5; year += 0.5) {
     System.out.println("蔡徐雞開始練習(xí)唱跳rap:已練習(xí)" + year + "年");
     try {
      Thread.sleep(288);
     } catch (InterruptedException e) {
      e.printStackTrace();
     }
     //眾所周知,練習(xí)兩年半即可出道
     if (year == 2.5) {
      System.out.println("===========================>成功練習(xí)兩年半,出道?。?!");
      //喚醒等待中的線程
      condition.signal();
     }
    }
    //業(yè)務(wù)代碼執(zhí)行完后解鎖
    lock.unlock();
   });
   //線程B,練習(xí)打籃球
   Thread threadB = new Thread(() -> {
    //執(zhí)行業(yè)務(wù)代碼前上鎖
    lock.lock();
    while (true) {
     try {
      //讓線程等待,如果計數(shù)器為0的話,則立即執(zhí)行
      condition.await();
     } catch (InterruptedException e) {
      e.printStackTrace();
     }
     System.out.println("蔡徐老母雞開始練習(xí)打籃球");
     break;
    }
    //業(yè)務(wù)代碼執(zhí)行完后解鎖
    lock.unlock();
   });
   //注意,一定要先啟動B,不然會導(dǎo)致B永遠拿不到鎖
   threadB.start();
   threadA.start();
  }
 }

運行結(jié)果:

蔡徐雞開始練習(xí)唱跳rap:已練習(xí)0.5年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)1.0年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)1.5年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)2.0年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)2.5年
===========================>成功練習(xí)兩年半,出道!?。?br /> 蔡徐雞開始練習(xí)唱跳rap:已練習(xí)3.0年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)3.5年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)4.0年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)4.5年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)5.0年
蔡徐老母雞開始練習(xí)打籃球

效果和synchronized+wait() 和 notify()一樣一樣的

3、基于volatile

使用共享變量也能實現(xiàn),用volatile即可,原理就是多個線程共同監(jiān)聽同個變量,根據(jù)變量的值變化來執(zhí)行對應(yīng)的任務(wù),此處volatile的作用就是讓其它線程能即時感知變量值的改變

public class volatileTest {
 //定義一個共享變量,注意,必須用volatile修飾
 static volatile boolean flag = false;
 //定義個year,用來記錄某明星練習(xí)打籃球的年數(shù)
 private static double year;

 public static void main(String[] args) {
  //線程A,練習(xí)唱跳rap
  Thread threadA = new Thread(() -> {
   while (true) {
    if (!flag) {
     for (year = 0.5; year <= 5; year += 0.5) {
      System.out.println("蔡徐雞開始練習(xí)唱跳rap:已練習(xí)" + year + "年");
      try {
       Thread.sleep(288);
      } catch (InterruptedException e) {
       e.printStackTrace();
      }
      //眾所周知,練習(xí)兩年半即可出道
      if (year == 2.5) {
       System.out.println("===========================>成功練習(xí)兩年半,出道?。?!");
       year = 0.5;
       flag = true;
       break;
      }
     }
    }
   }
  });
  //線程B,練習(xí)打籃球
  Thread threadB = new Thread(() -> {
   while (true) {
    if (flag) {
     System.out.println("蔡徐老母雞開始練習(xí)打籃球");
     break;
    }
   }
  });
  // 啟動線程
  threadA.start();
  threadB.start();
 }
}

運行結(jié)果:

蔡徐雞開始練習(xí)唱跳rap:已練習(xí)0.5年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)1.0年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)1.5年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)2.0年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)2.5年
===========================>成功練習(xí)兩年半,出道?。。?br /> 蔡徐老母雞開始練習(xí)打籃球

基于CountDownLatch

CountDownLatch是JUC包下的一個并發(fā)編程工具,主要有兩個方法,countDown和await,CountDownLatch底層維護了一個計數(shù)器,在實例化的時候設(shè)置,當(dāng)調(diào)用countDown方法時,計數(shù)器減一,如果計數(shù)器在減一前已經(jīng)為0,那么什么都不會發(fā)生,如果減一后變成0,則喚醒所有等待的線程;await方法會使當(dāng)前線程等待,直到計數(shù)器為0

public class CountDownLatchTest {
 //定義個year,用來記錄某明星練習(xí)打籃球的年數(shù)
 private static double year;

 public static void main(String[] args) {
  CountDownLatch latch = new CountDownLatch(1);
  //線程A,練習(xí)唱跳rap
  Thread threadA = new Thread(() -> {
   for (year = 0.5; year <= 5; year += 0.5) {
    System.out.println("蔡徐雞開始練習(xí)唱跳rap:已練習(xí)" + year + "年");
    try {
     Thread.sleep(288);
    } catch (InterruptedException e) {
     e.printStackTrace();
    }
    //眾所周知,練習(xí)兩年半即可出道
    if (year == 2.5) {
     System.out.println("===========================>成功練習(xí)兩年半,出道?。?!");
     //計數(shù)器減一
     latch.countDown();
    }
   }
  });
  //線程B,練習(xí)打籃球
  Thread threadB = new Thread(() -> {
   while (true) {
    try {
     //讓線程等待,如果計數(shù)器為0的話,則立即執(zhí)行
     latch.await();
    } catch (InterruptedException e) {
     e.printStackTrace();
    }
    System.out.println("蔡徐老母雞開始練習(xí)打籃球");
    break;
   }
  });
  // 啟動線程
  threadA.start();
  threadB.start();
 }
}

運行結(jié)果:

蔡徐雞開始練習(xí)唱跳rap:已練習(xí)0.5年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)1.0年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)1.5年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)2.0年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)2.5年
===========================>成功練習(xí)兩年半,出道!?。?br /> 蔡徐雞開始練習(xí)唱跳rap:已練習(xí)3.0年
蔡徐老母雞開始練習(xí)打籃球
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)3.5年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)4.0年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)4.5年
蔡徐雞開始練習(xí)唱跳rap:已練習(xí)5.0年

如果你多運行幾次,你會發(fā)現(xiàn)線程B執(zhí)行的時機是隨機的,但永遠在計數(shù)器為0后才開始執(zhí)行,也就是說計數(shù)器為0后,線程A和線程B誰搶到鎖就誰執(zhí)行

文中所有demo都是復(fù)制即可運行,大家還是要多動手,家里有條件的都用idea跑一跑,沒條件的可以用手抄

總結(jié)

到此這篇關(guān)于java線程間通訊的文章就介紹到這了,更多相關(guān)java線程間通訊內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評論