最近在拾遺JS基礎(chǔ),看到Promise拆座,總覺得和Java中的Future有聯(lián)系主巍,放在一塊對(duì)比總結(jié)一下,加深記憶挪凑。
異步編程
異步編程是為了提高效率孕索。其實(shí)這是一種在工作、生活都中非常常見的策略躏碳。
比如搞旭,我們跟別人交待任務(wù),拜托別人完成了通知我們菇绵,這就是一種異步肄渗。我們不需要等到別人完成了再繼續(xù)我們自己的工作,而只要等待別人的通知即可咬最。我們和別人可以并行的做事翎嫡。
既然是并行,那就需要至少兩個(gè)干活的人永乌,在計(jì)算機(jī)里面通常意味著多線程惑申,問題來了JS是單線程的,那是怎么做到的呢翅雏?
答案是:雖然JS運(yùn)行在瀏覽器中圈驼,是單線程的,但瀏覽器不是單線程的望几。瀏覽器中很多異步行為都是由瀏覽器新開一個(gè)線程去完成绩脆。javascript引擎線程是瀏覽器多個(gè)線程中的一個(gè),它本身是單線程的橄抹。瀏覽器還包括很多其他線程靴迫,如界面渲染線程,瀏覽器事件觸發(fā)線程害碾,Http請(qǐng)求線程等矢劲。所以,所謂的javascript是單線程的慌随,是指javascript運(yùn)行在瀏覽器中是單線程的,叫做javascript引擎線程躺同。
順帶提一下同步異步和阻塞阁猜、非阻塞的區(qū)別。這也是一個(gè)老生常談的問題:
同步/異步是從行為角度描述事物的蹋艺,而阻塞和非阻塞描述的當(dāng)前事物的狀態(tài)(等待調(diào)用結(jié)果時(shí)的狀態(tài))剃袍。
具體可以看這個(gè):『怎樣理解阻塞非阻塞與同步異步的區(qū)別?』
Future
好了下面說一下future捎谨。這個(gè)是是java的concurrent并發(fā)包里面的提供的類民效,他和Callable類常常一塊使用憔维。Future 表示異步計(jì)算的結(jié)果。其用于獲取線程池執(zhí)行callable后的結(jié)果畏邢,這個(gè)結(jié)果封裝為Future類业扒。
值得一提的是,由于Future類獲取返回結(jié)果的get方法是阻塞的舒萎,jdk1.8中加入completableFuture使得Future可以注冊(cè)監(jiān)聽器從而實(shí)現(xiàn)類似JS中的callback功能程储。這也是異步編程的推薦方式:回調(diào)。
Promise
回調(diào)是一種我們推崇的異步調(diào)用方式臂寝,但也會(huì)遇到問題章鲤,也就是回調(diào)的嵌套。當(dāng)需要多個(gè)異步回調(diào)一起書寫時(shí)咆贬,就會(huì)出現(xiàn)下面的代碼 (以 js 為例):
asyncFunc1(opt, (...args1) => {
asyncFunc2(opt, (...args2) => {
asyncFunc3(opt, (...args3) => {
asyncFunc4(opt, (...args4) => {
// some operation
});
});
});
});
這樣的代碼不易讀败徊,嵌套太深修改也麻煩。于是 ES6 提出了 Promise 模式來解決回調(diào)地獄的問題掏缎。
Promise 是一個(gè) ECMAScript 6 提供的類集嵌,目的是更加優(yōu)雅地書寫復(fù)雜的異步任務(wù)。
new Promise(function (resolve, reject) {
console.log(1111);
resolve(2222);
}).then(function (value) {
console.log(value);
return 3333;
}).then(function (value) {
console.log(value);
throw "An error";
}).catch(function (err) {
console.log(err);
});
Future和Promise的區(qū)別
通常御毅,F(xiàn)uture 和 Promise 兩個(gè)名詞在這個(gè)并發(fā)模型里是混用的根欧,無論是 Future 還是 Promise 都指代一個(gè)用于取得計(jì)算結(jié)果的代理或者計(jì)算結(jié)果的容器。不同之處是Future代表一個(gè)只讀的還不存在的結(jié)果的容器端蛆。而Promise是一次性可寫的凤粗。但Promise也實(shí)現(xiàn)了Future的接口。
Java 1.8 版本引入了 CompletableFuture 對(duì)象今豆,這個(gè)對(duì)象實(shí)現(xiàn)的 CompletionStage 接口定義了一系列的流暢的編程接口以支持靈活擴(kuò)展異步計(jì)算嫌拣。此外,CompletableFuture 支持手動(dòng)設(shè)定計(jì)算的結(jié)果呆躲,這就實(shí)現(xiàn)了 Promise 寫入的語義异逐。此前,我們只能通過立即返回的 Callable 來繞過插掂』艺埃總的來說,CompletableFuture 是一個(gè) Future 和 Promise 語義兼具的重載對(duì)象辅甥。
參考
https://www.zhihu.com/question/31982417/answer/54136684
https://www.cnkirito.moe/future-and-promise/
https://stackoverflow.com/questions/14541975/whats-the-difference-between-a-future-and-a-promise#:~:text=Futures%20and%20promises%20are%20pretty,written%20(normally%20only%20once).&text=The%20result%20of%20the%20future,result%20of%20an%20asynchronous%20computation.
https://zhuanlan.zhihu.com/p/71824799