協(xié)程必知必會(huì)-系列1-協(xié)程是什么

協(xié)程(Coroutine)是什么荔仁?

協(xié)程就是用戶態(tài)的線程育特。

這樣解釋可能過于抽象,讓我們先來回顧一下疫萤,另外2個(gè)更常見的概念颂跨,進(jìn)程(Process)與線程(Thread)。

「進(jìn)程是操作系統(tǒng)分配資源的基本單位」扯饶,只有在進(jìn)程內(nèi)才可以進(jìn)行內(nèi)存分配釋放恒削、文件讀寫、網(wǎng)卡數(shù)據(jù)的接收與發(fā)送等的資源操作尾序。

「線程是操作系統(tǒng)調(diào)度的基本單位」钓丰。

進(jìn)程和線程的狀態(tài)對應(yīng)用程序透明,并且在內(nèi)核態(tài)中完成調(diào)度每币。

協(xié)程對應(yīng)用程序來說是有狀態(tài)的携丁,需要應(yīng)用程序自行在用戶態(tài)中完成協(xié)程的調(diào)度。

image.png

協(xié)程解決什么問題兰怠?

其實(shí)協(xié)程這個(gè)概念很早就被提出來了梦鉴,到了互聯(lián)網(wǎng)高速發(fā)展的階段李茫,才被重視起來。

互聯(lián)網(wǎng)上但凡熱門的應(yīng)用肥橙,都至少有成百萬魄宏、千萬甚至是上億的用戶在使用,服務(wù)端同一時(shí)間需要處理大量用戶的請求存筏。

服務(wù)端需要具備處理高并發(fā)請求的能力娜庇,快速響應(yīng)用戶的請求。

互聯(lián)網(wǎng)的服務(wù)端基本上執(zhí)行都是「IO密集型的任務(wù)」方篮,而傳統(tǒng)的多進(jìn)程、多線程并發(fā)模型并不能高效的利用cpu励负,它們在遇到IO阻塞的時(shí)候藕溅,當(dāng)前的進(jìn)程或者線程就會(huì)被掛起,并發(fā)處理請求的能力有限继榆。

雖然可以使用「IO多路復(fù)用 + Reactor模型」來實(shí)現(xiàn)高并發(fā)巾表,但是這種方案下,業(yè)務(wù)代碼中充斥著很多的「異步回調(diào)函數(shù)」略吨,開發(fā)人員「心智負(fù)擔(dān)很重集币,代碼很難維護(hù)」。

引入?yún)f(xié)程后可以很好的解決這個(gè)問題翠忠,「IO阻塞時(shí)當(dāng)前協(xié)程自動(dòng)讓出執(zhí)行權(quán)鞠苟,等IO就緒時(shí)再恢復(fù)之前被掛起協(xié)程的執(zhí)行」。這樣就可以在業(yè)務(wù)層采用「同步編碼」秽之,而最后是「異步執(zhí)行」的效果当娱。在降低心智負(fù)擔(dān)的同時(shí),也能提供高性能的服務(wù)考榨。

協(xié)程該如何使用跨细?

一個(gè)協(xié)程庫的實(shí)現(xiàn),至少需要提供3個(gè)API河质,它們分別是:協(xié)程創(chuàng)建(CoroutineCreate)冀惭、協(xié)程喚醒(CoroutineResume),協(xié)程讓出(CoroutineYield)掀鹅。

為了更好的讓大家掌握協(xié)程的概念散休,我自己使用C++11實(shí)現(xiàn)了一個(gè)協(xié)程庫,并把它開源在github上淫半,地址鏈接為:https://github.com/wanmuc/MyCoroutine

現(xiàn)在讓我們來看看溃槐,該如何使用上面的協(xié)程庫,在協(xié)程中打印出”hello world“科吭,示例代碼如下所示昏滴。

#include "mycoroutine.h"
#include <iostream>

using namespace std;
using namespace MyCoroutine;

void HelloWorld(Schedule &schedule) {
  cout << "hello ";
  schedule.CoroutineYield();
  cout << "world" << endl;
}

int main() {
  // 創(chuàng)建一個(gè)協(xié)程調(diào)度對象猴鲫,并自動(dòng)生成大小為1024的協(xié)程池
  Schedule schedule(1024);
  // 創(chuàng)建一個(gè)從協(xié)程,并手動(dòng)調(diào)度
  int32_t cid = schedule.CoroutineCreate(HelloWorld, ref(schedule));
  schedule.CoroutineResume(cid);
  schedule.CoroutineResume(cid);
  return 0;
}

在上述代碼中谣殊,我們先創(chuàng)建了一個(gè)協(xié)程調(diào)度對象schedule拂共,它會(huì)自動(dòng)創(chuàng)建對應(yīng)協(xié)程池。Schedule類的成員函數(shù)CoroutineCreate用于創(chuàng)建協(xié)程姻几,成員函數(shù)CoroutineResume用于恢復(fù)協(xié)程的執(zhí)行宜狐,成員函數(shù)CoroutineYield用于協(xié)程讓出執(zhí)行權(quán)。

在main函數(shù)中蛇捌,創(chuàng)建完協(xié)程之后抚恒,調(diào)用CoroutineResume來啟動(dòng)協(xié)程的執(zhí)行,協(xié)程的啟動(dòng)HelloWorld被執(zhí)行络拌,打印完”hello “之后俭驮,協(xié)程主動(dòng)調(diào)用CoroutineYield函數(shù),讓出執(zhí)行權(quán)春贸。

進(jìn)程回到main函數(shù)中執(zhí)行混萝,再次調(diào)用CoroutineResume函數(shù),恢復(fù)協(xié)程的執(zhí)行萍恕,協(xié)程從上一次被中斷的地方繼續(xù)執(zhí)行逸嘀,打印出"world",然后協(xié)程執(zhí)行完畢并退出允粤。

協(xié)程退出之后崭倘,進(jìn)程回到main函數(shù)中繼續(xù)執(zhí)行,main執(zhí)行return語句类垫,整個(gè)進(jìn)程退出執(zhí)行绳姨。

協(xié)程本質(zhì)上是在一個(gè)進(jìn)程中,「創(chuàng)建多個(gè)調(diào)用棧幀阔挠,并在不同的調(diào)用棧幀之間切換的執(zhí)行」飘庄。在上面的例子中,main函數(shù)和HelloWorld函數(shù)就是兩個(gè)獨(dú)立的調(diào)用棧幀购撼。

如果還不能很好理解整個(gè)調(diào)度過程跪削,可以參考下圖。

image.png

本文為大廠后端技術(shù)專家萬木春原創(chuàng)文章迂求。作者更多技術(shù)干貨碾盐,見下方的書籍。


image.png
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末揩局,一起剝皮案震驚了整個(gè)濱河市毫玖,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖付枫,帶你破解...
    沈念sama閱讀 216,470評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件烹玉,死亡現(xiàn)場離奇詭異,居然都是意外死亡阐滩,警方通過查閱死者的電腦和手機(jī)二打,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,393評論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來掂榔,“玉大人继效,你說我怎么就攤上這事∽盎瘢” “怎么了瑞信?”我有些...
    開封第一講書人閱讀 162,577評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長穴豫。 經(jīng)常有香客問我喧伞,道長,這世上最難降的妖魔是什么绩郎? 我笑而不...
    開封第一講書人閱讀 58,176評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮翁逞,結(jié)果婚禮上肋杖,老公的妹妹穿的比我還像新娘。我一直安慰自己挖函,他們只是感情好状植,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,189評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著怨喘,像睡著了一般津畸。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上必怜,一...
    開封第一講書人閱讀 51,155評論 1 299
  • 那天肉拓,我揣著相機(jī)與錄音,去河邊找鬼梳庆。 笑死暖途,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的膏执。 我是一名探鬼主播驻售,決...
    沈念sama閱讀 40,041評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼更米!你這毒婦竟也來了欺栗?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,903評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎迟几,沒想到半個(gè)月后消请,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,319評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡瘤旨,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,539評論 2 332
  • 正文 我和宋清朗相戀三年梯啤,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片存哲。...
    茶點(diǎn)故事閱讀 39,703評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡因宇,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出祟偷,到底是詐尸還是另有隱情察滑,我是刑警寧澤,帶...
    沈念sama閱讀 35,417評論 5 343
  • 正文 年R本政府宣布修肠,位于F島的核電站贺辰,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏嵌施。R本人自食惡果不足惜饲化,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,013評論 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望吗伤。 院中可真熱鬧吃靠,春花似錦、人聲如沸足淆。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,664評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽巧号。三九已至族奢,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間丹鸿,已是汗流浹背越走。 一陣腳步聲響...
    開封第一講書人閱讀 32,818評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留靠欢,地道東北人弥姻。 一個(gè)月前我還...
    沈念sama閱讀 47,711評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像掺涛,于是被迫代替她去往敵國和親庭敦。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,601評論 2 353

推薦閱讀更多精彩內(nèi)容