日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Java異步調用轉同步方法實例詳解

瀏覽:131日期:2022-08-30 13:02:27

先說一下對異步和同步的理解:

同步調用:調用方在調用過程中,持續等待返回結果。

異步調用:調用方在調用過程中,不直接等待返回結果,而是執行其他任務,結果返回形式通常為回調函數。

其實,兩者的區別還是很明顯的,這里也不再細說,我們主要來說一下Java如何將異步調用轉為同步。換句話說,就是需要在異步

調用過程中,持續阻塞至獲得調用結果。

不賣關子,先列出五種方法,然后一一舉例說明:

使用wait和notify方法 使用條件鎖 Future 使用CountDownLatch 使用CyclicBarrier

0.構造一個異步調用

首先,寫demo需要先寫基礎設施,這里的話主要是需要構造一個異步調用模型。異步調用類:

public class AsyncCall { private Random random = new Random(System.currentTimeMillis()); private ExecutorService tp = Executors.newSingleThreadExecutor(); //demo1,2,4,5調用方法 public void call(BaseDemo demo){ new Thread(()->{ long res = random.nextInt(10); try {Thread.sleep(res*1000); } catch (InterruptedException e) {e.printStackTrace(); } demo.callback(res); }).start(); } //demo3調用方法 public Future<Long> futureCall(){ return tp.submit(()-> { long res = random.nextInt(10); try {Thread.sleep(res*1000); } catch (InterruptedException e) {e.printStackTrace(); } return res; }); } public void shutdown(){ tp.shutdown(); }}

我們主要關心call方法,這個方法接收了一個demo參數,并且開啟了一個線程,在線程中執行具體的任務,并利用demo的callback方法進行回調函數的調用。大家注意到了這里的返回結果就是一個[0,10)的長整型,并且結果是幾,就讓線程sleep多久——這主要是為了更好地觀察實驗結果,模擬異步調用過程中的處理時間。至于futureCall和shutdown方法,以及線程池tp都是為了demo3利用Future來實現做準備的。

demo的基類:

public abstract class BaseDemo { protected AsyncCall asyncCall = new AsyncCall(); public abstract void callback(long response); public void call(){ System.out.println('發起調用'); asyncCall.call(this); System.out.println('調用返回'); }}

BaseDemo非常簡單,里面包含一個異步調用類的實例,另外有一個call方法用于發起異步調用,當然還有一個抽象方法callback需要每個demo去實現的——主要在回調中進行相應的處理來達到異步調用轉同步的目的。

1. 使用wait和notify方法

這個方法其實是利用了鎖機制,直接貼代碼:

public class Demo1 extends BaseDemo{ private final Object lock = new Object(); @Override public void callback(long response) { System.out.println('得到結果'); System.out.println(response); System.out.println('調用結束'); synchronized (lock) { lock.notifyAll(); } } public static void main(String[] args) { Demo1 demo1 = new Demo1(); demo1.call(); synchronized (demo1.lock){ try {demo1.lock.wait(); } catch (InterruptedException e) {e.printStackTrace(); } } System.out.println('主線程內容'); }}

可以看到在發起調用后,主線程利用wait進行阻塞,等待回調中調用notify或者notifyAll方法來進行喚醒。注意,和大家認知的一樣,這里wait和notify都是需要先獲得對象的鎖的。在主線程中最后我們打印了一個內容,這也是用來驗證實驗結果的,如果沒有wait和notify,主線程內容會緊隨調用內容立刻打印;而像我們上面的代碼,主線程內容會一直等待回調函數調用結束才會進行打印。

沒有使用同步操作的情況下,打印結果:

發起調用調用返回主線程內容得到結果1調用結束

而使用了同步操作后:

發起調用調用返回得到結果9調用結束主線程內容

2. 使用條件鎖

和方法一的原理類似:

public class Demo2 extends BaseDemo { private final Lock lock = new ReentrantLock(); private final Condition con = lock.newCondition(); @Override public void callback(long response) { System.out.println('得到結果'); System.out.println(response); System.out.println('調用結束'); lock.lock(); try { con.signal(); }finally { lock.unlock(); } } public static void main(String[] args) { Demo2 demo2 = new Demo2(); demo2.call(); demo2.lock.lock(); try { demo2.con.await(); } catch (InterruptedException e) { e.printStackTrace(); }finally { demo2.lock.unlock(); } System.out.println('主線程內容'); }}

基本上和方法一沒什么區別,只是這里使用了條件鎖,兩者的鎖機制有所不同。

3. Future

使用Future的方法和之前不太一樣,我們調用的異步方法也不一樣。

public class Demo3{ private AsyncCall asyncCall = new AsyncCall(); public Future<Long> call(){ Future<Long> future = asyncCall.futureCall(); asyncCall.shutdown(); return future; } public static void main(String[] args) { Demo3 demo3 = new Demo3(); System.out.println('發起調用'); Future<Long> future = demo3.call(); System.out.println('返回結果'); while (!future.isDone() && !future.isCancelled()); try { System.out.println(future.get()); } catch (InterruptedException e) { e.printStackTrace(); } catch (ExecutionException e) { e.printStackTrace(); } System.out.println('主線程內容'); }}

我們調用futureCall方法,方法中會想線程池tp提交一個Callable,然后返回一個Future,這個Future就是我們demo3中call中得到的,得到future對象之后就可以關閉線程池啦,調用asyncCall的shutdown方法。關于關閉線程池這里有一點需要注意,我們回過頭來看看asyncCall的shutdown方法:

public void shutdown(){ tp.shutdown(); }

發現只是簡單調用了線程池的shutdown方法,然后我們說注意的點,這里最好不要用tp的shutdownNow方法,該方法會試圖去中斷線程中中正在執行的任務;也就是說,如果使用該方法,有可能我們的future所對應的任務將被中斷,無法得到執行結果。然后我們關注主線程中的內容,主線程的阻塞由我們自己來實現,通過future的isDone和isCancelled來判斷執行狀態,一直到執行完成或被取消。隨后,我們打印get到的結果。

4. 使用CountDownLatch

使用CountDownLatch或許是日常編程中最常見的一種了,也感覺是相對優雅的一種:

public class Demo4 extends BaseDemo{ private final CountDownLatch countDownLatch = new CountDownLatch(1); @Override public void callback(long response) { System.out.println('得到結果'); System.out.println(response); System.out.println('調用結束'); countDownLatch.countDown(); } public static void main(String[] args) { Demo4 demo4 = new Demo4(); demo4.call(); try { demo4.countDownLatch.await(); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println('主線程內容'); }}

正如大家平時使用的那樣,此處在主線程中利用CountDownLatch的await方法進行阻塞,在回調中利用countDown方法來使得其他線程await的部分得以繼續運行。

當然,這里和demo1和demo2中都一樣,主線程中阻塞的部分,都可以設置一個超時時間,超時后可以不再阻塞。

5. 使用CyclicBarrier

public class Demo5 extends BaseDemo{ private CyclicBarrier cyclicBarrier = new CyclicBarrier(2); @Override public void callback(long response) { System.out.println('得到結果'); System.out.println(response); System.out.println('調用結束'); try { cyclicBarrier.await(); } catch (InterruptedException e) { e.printStackTrace(); } catch (BrokenBarrierException e) { e.printStackTrace(); } } public static void main(String[] args) { Demo5 demo5 = new Demo5(); demo5.call(); try { demo5.cyclicBarrier.await(); } catch (InterruptedException e) { e.printStackTrace(); } catch (BrokenBarrierException e) { e.printStackTrace(); } System.out.println('主線程內容'); }}

大家注意一下,CyclicBarrier和CountDownLatch僅僅只是類似,兩者還是有一定區別的。比如,一個可以理解為做加法,等到加到這個數字后一起運行;一個則是減法,減到0繼續運行。一個是可以重復計數的;另一個不可以等等等等。

另外,使用CyclicBarrier的時候要注意兩點。第一點,初始化的時候,參數數字要設為2,因為異步調用這里是一個線程,而主線程是一個線程,兩個線程都await的時候才能繼續執行,這也是和CountDownLatch區別的部分。第二點,也是關于初始化參數的數值的,和這里的demo無關,在平時編程的時候,需要比較小心,如果這個數值設置得很大,比線程池中的線程數都大,那么就很容易引起死鎖了。

總結

綜上,就是本次需要說的幾種方法了。事實上,所有的方法都是同一個原理,也就是在調用的線程中進行阻塞等待結果,而在回調中函數中進行阻塞狀態的解除。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網。

標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
亚洲视频二区| 免费观看亚洲| 日韩影院精彩在线| 99视频精品| 爽好久久久欧美精品| 久久影视一区| 精品一区三区| 久久av在线| 日韩一区二区三区精品| 亚州精品视频| 欧美色综合网| 九九久久国产| 亚洲成人不卡| 在线成人直播| 蜜臀91精品一区二区三区| 日韩欧美在线精品| 国产精品一级| 福利一区在线| 99免费精品| 亚洲一区日本| 日韩中文字幕| 国产精品国产三级在线观看| 精品亚洲成人| 日韩中文视频| 亚洲综合另类| 国产剧情在线观看一区| 福利一区二区免费视频| 久久久久久久久久久妇女| 91成人精品| 亚洲日本国产| 国内精品伊人| 国产农村妇女精品一区二区| 国产九一精品| 伊人精品一区| 日韩三级精品| 国产中文字幕一区二区三区| 欧美日韩精品免费观看视完整| 黄色亚洲精品| 国产乱人伦精品一区| 久久婷婷丁香| 日本精品另类| 成人日韩在线| 日韩成人精品一区二区三区| 极品av在线| 蜜桃视频一区二区三区在线观看| 国产精品一区三区在线观看| 精品日韩视频| 日韩国产欧美三级| 欧美三级精品| 日韩精品久久理论片| 久久青青视频| 蜜臀久久久99精品久久久久久| 精品亚洲自拍| 美国三级日本三级久久99| 老司机精品在线| 国产美女精品| 国产传媒在线观看| 一区二区国产精品| а√在线中文在线新版| 亚洲毛片网站| 久久精品国内一区二区三区水蜜桃| 亚洲18在线| 91精品国产福利在线观看麻豆| 69精品国产久热在线观看| 久久亚洲在线| 精品视频一区二区三区在线观看| 婷婷成人基地| 精品美女在线视频| 蜜臀精品久久久久久蜜臀| 亚洲精品永久免费视频| 日韩高清欧美激情| 黄色国产精品| 亚洲永久av| 日韩欧美中文字幕电影| 国产综合精品| 国产精品毛片久久| 国产欧美日韩精品高清二区综合区| 99xxxx成人网| 成人精品亚洲| 精品久久在线| 日韩精品一级| 欧美~级网站不卡| 精品国产一区二区三区av片| 日韩激情综合| 一区二区三区国产盗摄| 午夜电影亚洲| 日韩欧美少妇| 福利精品一区| 久久久久久亚洲精品美女| 亚洲日本久久| 妖精视频成人观看www| 日韩欧美视频专区| 黄色精品视频| 国产一区一一区高清不卡| 国产欧美日韩免费观看| 日本一区中文字幕| 每日更新成人在线视频| 在线视频观看日韩| 日本精品影院| 亚洲啊v在线| 久久久久久婷| 精品久久久久中文字幕小说| 日韩精品久久理论片| 美国三级日本三级久久99| 91久久在线| 久久九九电影| 99精品电影| 日韩精品诱惑一区?区三区| 欧美1区二区| 欧美综合社区国产| 亚洲久久一区| 视频一区欧美精品| 一区视频在线| 亚洲精品a级片| 欧美+日本+国产+在线a∨观看| 欧美日韩精品一区二区视频| 秋霞影视一区二区三区| 韩国精品主播一区二区在线观看| 国产福利片在线观看| 亚洲黄色免费av| 91亚洲人成网污www| 日韩深夜视频| 超碰成人av| 日韩在线高清| 日韩免费视频| 91精品国产成人观看| 成人福利av| 日韩精品2区| 久久久久国产| 99pao成人国产永久免费视频| 精品中文一区| 亚洲一区日韩在线| 日韩在线卡一卡二| 蜜臀精品久久久久久蜜臀| 日韩中文字幕亚洲一区二区va在线| 丝袜美腿成人在线| 婷婷亚洲成人| 国产欧美亚洲一区| 久久精品女人| 日韩网站中文字幕| 欧美日韩少妇| 国产农村妇女精品一二区| 噜噜噜躁狠狠躁狠狠精品视频 | 久久国产三级精品| 欧美精品中文| 国内精品伊人| 久久精品欧美一区| 精品1区2区3区4区| 亚洲九九精品| 国产亚洲精aa在线看| 麻豆国产精品777777在线| 精品一区二区三区中文字幕视频| 国产一区二区三区免费在线| 国产中文在线播放| 亚洲一本视频| 久热re这里精品视频在线6| 亚洲专区视频| 久久的色偷偷| 久久精品国语| 一区二区三区网站| 国产精品嫩草影院在线看| 久久99国产精品视频| 香蕉成人av| 亚洲一区二区三区无吗| 日韩精品乱码av一区二区| 精品国产一区二区三区av片| 国产尤物精品| 久久不射网站| 国产精品黄色片| 91精品国产自产在线观看永久∴| 蜜臀久久久99精品久久久久久| 国产精品**亚洲精品| 精精国产xxxx视频在线野外 | 欧美日韩国产综合网| 日韩欧美精品一区二区综合视频| 久久精品一区二区三区中文字幕| 久久久久国产| 欧美日韩a区| 欧美日韩精品免费观看视完整| 视频一区欧美日韩| 国产精品一卡| 精品一区免费| 国产精品22p| 国产精品美女久久久| 国产精品theporn| 国产一区视频在线观看免费| 日韩av网站免费在线| 亚洲www啪成人一区二区| 亚洲乱码一区| 中文字幕在线看片| 综合欧美亚洲| 久久91导航| 国产一卡不卡| 好吊日精品视频| 国产精品久久久久久久久久妞妞| 特黄毛片在线观看| 日韩国产精品久久久| 99久久亚洲精品蜜臀| 国产精品最新| 国产婷婷精品|