Java異步非阻塞編程的幾種方式總結(jié)
1 服務(wù)端執(zhí)行,最簡(jiǎn)單的同步調(diào)用方式:
- 缺陷:
- 服務(wù)端響應(yīng)之前,IO會(huì)阻塞在:
- java.net.SocketInputStream#socketRead0 的native方法上:
2 JDK NIO & Future java 1.5之后
- 優(yōu)點(diǎn):主線(xiàn)程可以不用等待IO響應(yīng),可以去做點(diǎn)其他的,比如說(shuō)再發(fā)送一個(gè)IO請(qǐng)求,可以等到一起返回;
- 缺點(diǎn):主線(xiàn)程在等待結(jié)果返回過(guò)程中依然需要等待,沒(méi)有根本解決此問(wèn)題;
3 使用Callback回調(diào)方式
- 優(yōu)點(diǎn):主線(xiàn)程完成發(fā)送請(qǐng)求后,再也不用關(guān)心這個(gè)邏輯,去執(zhí)行其他的邏輯;整個(gè)過(guò)程中已經(jīng)沒(méi)有線(xiàn)程阻塞;如 使用nio的EventLoopGroup中的線(xiàn)程執(zhí)行完所有邏輯;
- 缺點(diǎn):回調(diào)地獄;Callback hell ;代碼可讀性低、編寫(xiě)費(fèi)勁、容易出錯(cuò)
4 JDK 1.8 CompletableFuture
- 優(yōu)點(diǎn):解決Callback Hell的問(wèn)題,JDK 1.8中提供了CompletableFuture;每一個(gè)IO操作,均可以封裝為獨(dú)立的CompletableFuture,從而避免回調(diào)地獄。
- 實(shí)現(xiàn):將逆Callback邏輯,封裝成一個(gè)獨(dú)立的CompletableFuture,當(dāng)異步線(xiàn)程回調(diào)時(shí),調(diào)用 future.complete(T) ,將結(jié)果封裝;thenCompose銜接,whenComplete輸出;
- 小結(jié):這樣一來(lái),就完美解決回調(diào)地獄問(wèn)題,在主的邏輯中,看起來(lái)像是在同步的進(jìn)行編碼。
5 源碼舉例 測(cè)試+結(jié)果
import java.util.concurrent.CompletableFuture;
public class CompletableFutureTest {
private static CompletableFuture<String> invokeAFuture(String rawASource){
CompletableFuture<String> future = new CompletableFuture<>();
System.out.println("pre-do-invokeA");
try {
Thread.sleep(1000);
future.complete("invokeA "+rawASource+" result = skip");
}catch (Exception e){
}
return future;
}
private static CompletableFuture<String> invokeBFuture(String rawAResult){
CompletableFuture<String> future = new CompletableFuture<>();
System.out.println("pre-do-invokeB");
try {
Thread.sleep(1000);
future.complete("after A done result = "+rawAResult+", then invokeB result = success");
}catch (Exception e){
}
return future;
}
public static void main(String[] args) {
invokeAFuture("加油").thenCompose(aResult-> invokeBFuture(aResult)).whenComplete((resultB, throwable) ->{
if(throwable != null){
throwable.printStackTrace();
return;
}
System.out.println(resultB);
});
}
public static void main(String[] args) {
invokeAFuture("加油").thenCompose(CompletableFutureTest::invokeBFuture).whenComplete((resultB, throwable) ->{
if(throwable != null){
throwable.printStackTrace();
return;
}
System.out.println(resultB);
});
}
}
pre-do-invokeA
pre-do-invokeB
after A done result = invokeA 加油 result = skip, then invokeB result = success
6 小結(jié):
- 1 嘗試使用異步編程方式;
- 2 剖析內(nèi)部實(shí)現(xiàn)原理;
- 3 java9 juc 包有了更抽象的flow處理方式;
Java 異步編程最佳實(shí)踐
什么是異步?為什么要用它?
異步編程提供了一個(gè)非阻塞的,事件驅(qū)動(dòng)的編程模型。 這種編程模型利用系統(tǒng)中多核執(zhí)行任務(wù)來(lái)提供并行,因此提供了應(yīng)用的吞吐率。此處吞吐率是指在單位時(shí)間內(nèi)所做任務(wù)的數(shù)量。 在這種編程方式下, 一個(gè)工作單元將獨(dú)立于主應(yīng)用線(xiàn)程而執(zhí)行, 并且會(huì)將它的狀態(tài)通知調(diào)用線(xiàn)程:成功,處理中或者失敗。
我們需要異步來(lái)消除阻塞模型。其實(shí)異步編程模型可以使用同樣的線(xiàn)程來(lái)處理多個(gè)請(qǐng)求, 這些請(qǐng)求不會(huì)阻塞這個(gè)線(xiàn)程。想象一個(gè)應(yīng)用正在使用的線(xiàn)程正在執(zhí)行任務(wù), 然后等待任務(wù)完成才進(jìn)行下一步。 log框架就是一個(gè)很好的例子:典型地你想將異常和錯(cuò)誤日志記錄到一個(gè)目標(biāo)中, 比如文件,數(shù)據(jù)庫(kù)或者其它類(lèi)似地方。你不會(huì)讓你的程序等待日志寫(xiě)完才執(zhí)行,否則程序的響應(yīng)就會(huì)受到影響。 相反,如果對(duì)log框架的調(diào)用是異步地,應(yīng)用就可以并發(fā)執(zhí)行其它任務(wù)而無(wú)需等待。這是一個(gè)非阻塞執(zhí)行的例子。
為了在Java中實(shí)現(xiàn)異步,你需要使用Future 和 FutureTask, 它們位于java.util.concurrent包下. Future是一個(gè)接口而FutureTask是它的一個(gè)實(shí)現(xiàn)類(lèi)。實(shí)際上,如果在你的代碼中使用Future, 你的異步任務(wù)會(huì)立即執(zhí)行, 并且調(diào)用線(xiàn)程可以得到結(jié)果promise。
該做和不該做的
為了方便測(cè)試,你應(yīng)該在代碼中將功能從多線(xiàn)程中隔離出來(lái)。當(dāng)在Java中編寫(xiě)異步代碼時(shí),你應(yīng)該遵循異步模型,這樣調(diào)用線(xiàn)程就不會(huì)被阻塞。
注意構(gòu)造函數(shù)不能是異步的,你不應(yīng)該在構(gòu)造函數(shù)中調(diào)用異步方法。當(dāng)任務(wù)互相不依賴(lài)時(shí)異步方式尤其有用。當(dāng)調(diào)用任務(wù)依賴(lài)被調(diào)用任務(wù)時(shí)不應(yīng)該使用異步(譯者按:這對(duì)異步來(lái)說(shuō)無(wú)意義,因?yàn)闃I(yè)務(wù)上調(diào)用線(xiàn)程被阻塞了).
你應(yīng)該在異步方法中處理異常. 你不應(yīng)該為長(zhǎng)時(shí)間的task實(shí)現(xiàn)異常. 一個(gè)長(zhǎng)時(shí)間運(yùn)行的任務(wù),如果異步執(zhí)行的話(huà), 可能會(huì)比同步執(zhí)行耗費(fèi)更長(zhǎng)的時(shí)間, 因?yàn)檫\(yùn)行時(shí)要為異步執(zhí)行的方法執(zhí)行線(xiàn)程上下文的切換, 線(xiàn)程狀態(tài)的存儲(chǔ)等. 你也應(yīng)該注意同步的異常和異步的異常有所不同。
同步異常暗示 每次程序執(zhí)行到那個(gè)程序特殊狀態(tài)時(shí)就會(huì)拋出異常;異步異常的跟蹤則困難的多。
所以同步和異步異常暗示同步或異步代碼可能拋出異常
synchronous and asynchronous exceptions imply synchronous or asynchronous code in your program that might raise exceptions.
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Spring系列中的beanFactory與ApplicationContext
這篇文章主要介紹了Spring系列中的beanFactory與ApplicationContext,文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-09-09
Mybatis一對(duì)多查詢(xún)的兩種姿勢(shì)(值得收藏)
這篇文章主要給大家介紹了關(guān)于Mybatis一對(duì)多查詢(xún)的兩種姿勢(shì),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-05-05
spring cloud gateway中如何讀取請(qǐng)求參數(shù)
這篇文章主要介紹了spring cloud gateway中如何讀取請(qǐng)求參數(shù)的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07
詳解Java并發(fā)編程之volatile關(guān)鍵字
這篇文章主要為大家介紹了Java并發(fā)編程之volatile關(guān)鍵字,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2021-11-11
SpringBoot集成SwaggerUi以及啟動(dòng)時(shí)遇到的錯(cuò)誤
這篇文章主要介紹了SpringBoot集成SwaggerUi以及啟動(dòng)時(shí)遇到的錯(cuò)誤,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06
SpringBoot中MockMVC單元測(cè)試的實(shí)現(xiàn)
Mock是一種用于模擬和替換類(lèi)的對(duì)象的方法,以便在單元測(cè)試中獨(dú)立于外部資源進(jìn)行測(cè)試,本文主要介紹了SpringBoot中MockMVC單元測(cè)試的實(shí)現(xiàn),具有應(yīng)該的參考價(jià)值,感興趣的可以了解一下2024-02-02
httpclient staleConnectionCheckEnabled獲取連接流程解析
這篇文章主要為大家介紹了httpclient staleConnectionCheckEnabled獲取連接流程示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11
play for scala 實(shí)現(xiàn)SessionFilter 過(guò)濾未登錄用戶(hù)跳轉(zhuǎn)到登錄頁(yè)面
這篇文章主要介紹了play for scala 實(shí)現(xiàn)SessionFilter 過(guò)濾未登錄用戶(hù)跳轉(zhuǎn)到登錄頁(yè)面的相關(guān)資料,需要的朋友可以參考下2016-11-11
SpringBoot使用classfinal-maven-plugin插件加密Jar包的示例代碼
這篇文章給大家介紹了SpringBoot使用classfinal-maven-plugin插件加密Jar包的實(shí)例,文中通過(guò)代碼示例和圖文講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-02-02
Mybatis-Plus 搭建與使用入門(mén)(小結(jié))
Mybatis-Plus(簡(jiǎn)稱(chēng)MP)是一個(gè) Mybatis 的增強(qiáng)工具,這篇文章主要介紹了Mybatis-Plus 搭建與使用入門(mén)(小結(jié)),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-06-06

