<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
應用中通過框架傳送非同步命令時,不能立刻返回命令的執行結果,而是非同步返回命令的執行結果。
那麼,問題來了,針對應用中這種非同步呼叫,能不能像同步呼叫一樣立刻獲取到命令的執行結果,如何實現非同步轉同步?
首先,解釋下同步和非同步
對於非同步呼叫,呼叫的返回並不受呼叫者控制。
非同步轉同步主要實現思路:所有實現原理類似,是在發出呼叫的執行緒中進行阻塞等待結果,呼叫完成後通過回撥、設定共用狀態或通知進行阻塞狀態的解除,繼續執行後續操作。
通常,實現中,不會無限的等待,一般會設定一個超時時間,具體超時時間根據具體場景確定。
下面以回撥的方式介紹幾種常用實現非同步轉同步的方法:
採用輪詢與休眠重試機制,執行緒將反覆在休眠和測試狀態條件中之間切換,直到超時或者狀態條件滿足繼續向下執行。這種方式,超時時間控制不準確,sleep時間需要在響應性和CPU使用率之間進行權衡。
private static long MILLIS_OF_WAIT_TIME = 300000L;// 等待時間 5分鐘 private final Object lock = new Object(); //3.結果返回後進行回撥,解除阻塞 @Override public void callback(AsynResponse response){ synchronized(lock){ //設定狀態條件 } public Result getResult() throws ErrorCodeException { // 1.非同步呼叫 // 2.阻塞等待非同步響應 long future = System.currentTimeMillis() + MILLIS_OF_WAIT_TIME; long remaining = MILLIS_OF_WAIT_TIME;//剩餘等待時間 while(remaining > 0){ synchronized(lock){ if(狀態條件未滿足){ remaining = future - System.currentTimeMillis(); Thread.sleep(時間具體場景確定); } } ````} //4.超時或結果正確返回,對結果進行處理 return result; }
任意一個Java物件,都擁有一組監視器方法(wait、notify、notifyAll等方法),這些方法和synchronized同步關鍵字配合,可以實現等待/通知模式。但是使用wait/notify,使執行緒的阻塞/喚醒對執行緒本身來說是被動的,要準確的控制哪個執行緒是很困難的,所以是要麼隨機喚醒等待在條件佇列上一個執行緒(notify),要麼喚醒所有的(notifyAll,但是很低效)。當多個執行緒基於不同條件在同一條件佇列上等待時,如果使用notify而不是notifyAll,很容易導致訊號丟失的問題,所以必須謹慎使用wait/notify方法。
private static long MILLIS_OF_WAIT_TIME = 300000L;// 等待時間 5分鐘 private final Object lock = new Object(); //3.結果返回後進行回撥,解除阻塞 @Override public void callback(AsynResponse response){ synchronized(lock){ lock.notifyAll(); } public Result getResult() throws ErrorCodeException { // 1.非同步呼叫 // 2.阻塞等待非同步響應 long future = System.currentTimeMillis() + MILLIS_OF_WAIT_TIME; long remaining = MILLIS_OF_WAIT_TIME;//剩餘等待時間 synchronized(lock){ while(條件未滿足 && remaining > 0){ //被通知後要檢查條件 lock.wait(remaining); remaining = future - System.currentTimeMillis(); } ````} //4.超時或結果正確返回,對結果進行處理 return result; }
使用Lock的Condition佇列的實現方式和wait/notify方式類似,但是Lock支援多個Condition佇列,並且支援等待狀態中響應中斷。
private static long SECONDS_OF_WAIT_TIME = 300L;// 等待時間 5分鐘 private final Lock lock = new ReentrantLock(); private final Condition condition = lock.newCondition(); //3.結果返回後進行回撥,解除阻塞 @Override public void callback(AsynResponse response){ lock.lock();//這是前提 try { condition.signal(); }finally { lock.unlock(); } } public Result getResult() throws ErrorCodeException { // 1.非同步呼叫 // 2.阻塞等待非同步響應 lock.lock();//這是前提 try { condition.await(); } catch (InterruptedException e) { //TODO }finally { lock.unlock(); } //4.超時或結果正確返回,對結果進行處理 return result; }
使用CountDownLatch可以實現非同步轉同步,它好比計數器,在建立範例CountDownLatch物件的時候傳入數位,每使用一次 countDown() 方法計數減1,當數位減到0時, await()方法後的程式碼將可以執行,未到0之前將一直阻塞等待。
private static long SECONDS_OF_WAIT_TIME = 300L;// 等待時間 5分鐘 private final CountDownLatch countDownLatch = new CountDownLatch(1); //3.結果返回後進行回撥,解除阻塞 @Override public void callback(AsynResponse response){ countDownLatch.countDown(); } public Result getResult() throws ErrorCodeException { // 1.非同步呼叫 // 2.阻塞等待非同步響應 try { countDownLatch.await(SECONDS_OF_WAIT_TIME, TimeUnit.SECONDS); } catch (InterruptedException e) { //TODO } //4.超時或結果正確返回,對結果進行處理 return result; }
讓一組執行緒達到一個屏障(也可以叫同步點)時被阻塞,直到等待最後一個執行緒到達屏障時,屏障才開門,所有被屏障攔截的執行緒才會繼續執行。
每個執行緒通過呼叫await方法告訴CyclicBarrier我已經到達了屏障,然後當前的的執行緒被阻塞。
private static long SECONDS_OF_WAIT_TIME = 300L;// 等待時間 5分鐘 private final CountDownLatch cyclicBarrier= new CyclicBarrier(2);//設定屏障攔截的執行緒數為2 //3.結果返回後進行回撥,解除阻塞 @Override public void callback(AsynResponse response){ //我也到達屏障了,可以開門了 cyclicBarrier.await(); } public Result getResult() throws ErrorCodeException { // 1.非同步呼叫 // 2.阻塞等待非同步響應 try { //我到達屏障了,還沒開門,要等一等 cyclicBarrier.await(SECONDS_OF_WAIT_TIME, TimeUnit.SECONDS); } catch (InterruptedException e) { //TODO } //4.超時或結果正確返回,對結果進行處理 return result; }
CountDownLatch和CyclicBarrier實現類似,區別是CountDownLatch的計數器只能使用一次,而CyclicBarrier的計數器可以使用reset重置,
所以CyclicBarrier能處理更為複雜的業務場景。在非同步轉同步中,計數器不會重用,所以使用CountDownLatch實現更適合。
LockSupport定義了一組公共靜態方法,提供了最基本的執行緒阻塞和喚醒的方法。
private static long NANOS_OF_WAIT_TIME = 300000000L;// 等待時間 5分鐘 private final LockSupport lockSupport = new LockSupport(); //3.結果返回後進行回撥,解除阻塞 @Override public void callback(AsynResponse response){ lockSupport.unpark(); } public Result getResult() throws ErrorCodeException { // 1.非同步呼叫 // 2.阻塞等待非同步響應 try { lockSupport.parkNanos(NANOS_OF_WAIT_TIME); } catch (InterruptedException e) { //TODO } //4.超時或結果正確返回,對結果進行處理 return result; }
到此這篇關於詳解Java中非同步轉同步的六種方法的文章就介紹到這了,更多相關Java非同步轉同步內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45