Java線程協(xié)作的兩種方式小結(jié)
為什么線程之間需要協(xié)作
線程之間相互配合,完成某項工作,比如:一個線程修改了一個對象的值,而另一個線程感知到了變化,然后進行相應(yīng)的操作,整個過程開始于一個線程,而最終執(zhí)行又是另一個線程。前者是生產(chǎn)者,后者就是消費者,這種模式隔離了“做什么”(What)和“怎么做”(How)。簡單的辦法是讓消費者線程不斷地循環(huán)檢查變量是否符合預(yù)期,在while循環(huán)中設(shè)置不滿足的條件,如果條件滿足則退出while循環(huán),從而完成消費者的工作。這樣進行線程之間的協(xié)作卻存在如下2個問題:
(1)難以確保及時性。
(2)難以降低開銷。如果降低睡眠的時間,比如休眠1毫秒,這樣消費者能更加迅速地發(fā)現(xiàn)條件變化,但是卻可能消耗更多的處理器資源,造成了無端的浪費。
那么有沒有什么辦法可以解決以上2個問題呢?此時等待/通知機制毫不客氣的站出來說,都讓開,交給我,我能行!
介紹
Java中線程協(xié)作的最常見的兩種方式:利用Object.wait()、Object.notify()和使用Condition
方法一
Object中的wait、notify、notifyAll方法定義如下
- public final native void notify(); public final native void notifyAll(); public final native void wait(long timeout) throws InterruptedException;
- wait()、notify()和notifyAll()方法是本地方法,并且為final方法,無法被重寫
- 調(diào)用某個對象的wait()方法能讓當(dāng)前線程阻塞,并且當(dāng)前線程必須擁有此對象的monitor(即鎖)
- 調(diào)用某個對象的notify()方法能夠喚醒一個正在等待這個對象的monitor的線程,如果有多個線程都在等待這個對象的monitor,則只能喚醒其中一個線程
- 調(diào)用notifyAll()方法能夠喚醒所有正在等待這個對象的monitor的線程
- 之所以這三個方法聲明在Object類中是因為每個對象都擁有monitor(即鎖)
- 調(diào)用某個對象的wait()方法,當(dāng)前線程必須擁有這個對象的monitor(即鎖),因此調(diào)用wait()方法必須在同步塊或者同步方法中進行
示例
public class Test { public static Object object = new Object(); public static void main(String[] args) { Thread1 thread1 = new Thread1(); Thread2 thread2 = new Thread2(); thread1.start(); try { Thread.sleep(200); } catch (InterruptedException e) { e.printStackTrace(); } thread2.start(); } static class Thread1 extends Thread{ @Override public void run() { synchronized (object) { try { object.wait(); } catch (InterruptedException e) { } System.out.println("線程"+Thread.currentThread().getName()+"獲取到了鎖"); } } } static class Thread2 extends Thread{ @Override public void run() { synchronized (object) { object.notify(); System.out.println("線程"+Thread.currentThread().getName()+"調(diào)用了object.notify()"); } System.out.println("線程"+Thread.currentThread().getName()+"釋放了鎖"); } } }
運行結(jié)果
線程Thread-1調(diào)用了object.notify()
線程Thread-1釋放了鎖
線程Thread-0獲取到了鎖
方法二
- Condition是在java 1.5中才出現(xiàn)的,它用來替代傳統(tǒng)的Object的wait()、notify()實現(xiàn)線程間的協(xié)作,相比使用Object的wait()、notify(),使用Condition1的await()、signal()這種方式實現(xiàn)線程間協(xié)作更加安全和高效
- Condition是個接口,基本的方法就是await()和signal()方法
- Condition依賴于Lock接口,生成一個Condition的基本代碼是lock.newCondition()
- 調(diào)用Condition的await()和signal()方法,都必須在lock保護之內(nèi),就是說必須在lock.lock()和lock.unlock之間才可以使用
示例
public class Test { private int queueSize = 10; private PriorityQueue<Integer> queue = new PriorityQueue<Integer>(queueSize); private Lock lock = new ReentrantLock(); private Condition notFull = lock.newCondition(); private Condition notEmpty = lock.newCondition(); public static void main(String[] args) { Test test = new Test(); Producer producer = test.new Producer(); Consumer consumer = test.new Consumer(); producer.start(); consumer.start(); } class Consumer extends Thread{ @Override public void run() { consume(); } private void consume() { while(true){ lock.lock(); try { while(queue.size() == 0){ try { System.out.println("隊列空,等待數(shù)據(jù)"); notEmpty.await(); } catch (InterruptedException e) { e.printStackTrace(); } } queue.poll(); //每次移走隊首元素 notFull.signal(); System.out.println("從隊列取走一個元素,隊列剩余"+queue.size()+"個元素"); } finally{ lock.unlock(); } } } } class Producer extends Thread{ @Override public void run() { produce(); } private void produce() { while(true){ lock.lock(); try { while(queue.size() == queueSize){ try { System.out.println("隊列滿,等待有空余空間"); notFull.await(); } catch (InterruptedException e) { e.printStackTrace(); } } queue.offer(1); //每次插入一個元素 notEmpty.signal(); System.out.println("向隊列取中插入一個元素,隊列剩余空間:"+(queueSize-queue.size())); } finally{ lock.unlock(); } } } } }
到此這篇關(guān)于Java線程協(xié)作的兩種方式小結(jié)的文章就介紹到這了,更多相關(guān)Java線程協(xié)作內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
idea中mapper如何快速跳轉(zhuǎn)到xml插件
這篇文章主要介紹了idea中mapper如何快速跳轉(zhuǎn)到xml插件問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-05-05IntelliJ?IDEA?2021.3?正式發(fā)布之支持遠程開發(fā)、IDE故障排查等多項優(yōu)化改進
IntelliJ?IDEA?2021.3?正式發(fā)布:支持遠程開發(fā)、IDE故障排查等多項優(yōu)化改進問題,在這個版本中的遠程開發(fā)還不是一個正式版本,而是BETA版,但通過這個BETA版本,也可以體驗IDEA“遠程開發(fā)”給我們帶來的全新體驗2021-12-12vue+springboot項目上傳部署tomcat的方法實現(xiàn)
本文主要介紹了vue+springboot項目上傳部署tomcat的方法實現(xiàn),包括環(huán)境準備、配置調(diào)整以及部署步驟,文中通過圖文及示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-01-01Java中JFinal框架動態(tài)切換數(shù)據(jù)庫的方法
這篇文章主要介紹了Java中JFinal框架動態(tài)切換數(shù)據(jù)庫的方法,本文通過兩種方法結(jié)合示例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-03-03IDEA中將SpringBoot項目提交到git倉庫的方法步驟
本文主要介紹了IDEA中將SpringBoot項目提交到git倉庫的方法步驟,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-12-12