Unix 網絡 IO 模型: 同步異步, 傻傻分不清楚?

出處

阻塞 IO, 非阻塞 IO, 同步 IO, 異步 IO 這些術語相信有不少朋友都也不同程度的困惑吧? 我原來也是, 什么同步非阻塞 IO, 異步非阻塞 IO 的, 搞的頭都大了. 后來仔細讀了一遍
《UNIX 網絡編程卷一 套接字聯網 API(第三版)》的 6.2 章節, 終于把這些名詞搞懂了.

下面我以《UNIX 網絡編程卷一 套接字聯網 API(第三版)》的 6.2 章節的內容為準, 整理了一下各種網絡 IO 模型具體定義以及一些容易混淆的地方.

簡介

Unix 下有 5 種可用的 IO 模型, 分別是:

  • 阻塞式 I/O
  • 非阻塞式 I/O
  • I/O 復用(select 和 poll)
  • 信號驅動式 I/O (SIGIO)
  • 異步 I/O (POSIX 的 aio_系列函數)

阻塞式 I/O 模型

最流行的 IO 操作是阻塞式 IO(Blocking IO). 以 UDP 數據報套接字為例, 下圖是其阻塞 IO 的調用過程:

1462343854639.png

在上圖中, 進程調用 recvfrom, 其系統調用直到數據報返回并且被復制到應用進程的緩沖區中 或者發送錯誤時才返回. 因此進程在調用 recvfrom 開始到它返回的整段時間內都是被阻塞的.

非阻塞式 IO(Non-Blocking IO)

進程把一個套接字設置為非阻塞是在通知內核: 當調用線程所請求的 IO 操作需要調用線程休眠來等待操作完成時, 此時不要將調用線程休眠, 而是返回一個錯誤.

非阻塞式 IO

如上圖所示, 前三次調用 recvfrom 時, 沒有數據可返回, 因此內核轉而立即返回一個 EWOULDBLOCK 錯誤. 第四次調用 recvfrom 時, 已經有數據了, 此時, recvfrom 會阻塞住, 等待內核將數據賦值到應用進程的緩沖區中, 然后再返回.(注意, 當有數據時, recvfrom 是阻塞的, 它會等待內核將數據復制到應用進程的緩沖區后, 才返回).
當一個應用進程像這樣對一個非阻塞描述符循環調用 recvfrom 時, 我們稱之為輪詢(polling). 應用進程持續輪詢內核, 以查看某個操作是否完成, 這么做會消耗大量的 CPU 時間, 不過這種模型偶爾也會遇到, 通常是專門提供某一種功能的系統中才有.

IO 復用模型

有了 IO 復用(IO multiplexing), 我們就可以調用 select 或 poll, 阻塞在這兩個系統調用中的某一個之上, 而不是阻塞在真正的 IO 系統調用上. 例如:

IO 復用模型

如上圖所示, 當調用了 select 后, select 會阻塞住, 等待數據報套接字變為可讀. 當 select 返回套接字可讀這一條件時, 我們就可以調用 recvfrom 把所讀取的數據報復制到應用進程緩沖區.
對比阻塞式 IO, IO 復用模型優勢并不明顯, 并且從使用方式來說, IO 復用模型還需要多調用一次 select, 因此從易用性上來說, 比阻塞式 IO 還略有不足. 不過 select 的殺手锏在于它可以監聽多個文件描述符, 大大減小了阻塞線程的個數.

信號驅動 IO 模型

信號驅動 IO 模型

信號驅動模型如上圖所示. 當文件描述符就緒時, 我們可以讓內核以信號的方式通知我們.
我們首先需要開啟套接字的信號驅動式 IO 功能, 并通過 sigaction 系統調用安裝一個信號處理函數. sigaction 系統調用是異步的, 它會立即返回. 當有數據時, 內核會給此進程發送一個 SIGIO 信號, 進而我們的信號處理函數就會被執行, 我們就可以在這個函數中調用 recvfrom 讀取數據.

異步 IO 模型

異步 IO (asynchronous IO) 由 POSIX 規范定義, 在 POSIX 中定義了若干個異步 IO 的操作函數. 這個函數的工作原理是: 告知內核啟動某個動作, 并讓內核在整個操作(包括將數據從內核復制到應用進程緩沖區)完成后通知我們的應用進程.
異步 IO 模型和信號驅動的 IO 模型的主要區別在于: 信號驅動 IO 是由內核通知我們何時可以啟動一個 IO 操作, 而異步 IO 模型是由內核通知我們 IO 操作何時完成.
異步 IO 模型的操作過程如圖所示:

異步 IO 模型

當我們調用 aio_read 函數時(POSIX 異步 IO 函數以 aio_或 lio_ 開頭), 給內核傳遞描述符, 緩沖區指針, 緩沖區大小(和 read 相同的三個參數) 和文件偏移(以 lseek 類似), 并告訴內核當整個操作完成時如何通知應用進程. 該系統調用立即返回, 而且在等待 IO 完成期間, 應用進程不被阻塞.

各種 IO 模型的比較

各種 IO 模型的比較

如圖所示, 上述五中 IO 模型中, 前四種模型(阻塞 IO, 非阻塞 IO, IO 復用, 信號驅動 IO)的主要區別在于第一階段, 因為他們的第二階段是一樣的: 在數據從內核復制到調用者的緩沖區期間, 進程阻塞于 recvfrom 調用. 而第五種, 即異步 IO 模型中, 兩個階段都不需要應用進程處理, 內核為我們處理好了數據的等待和數據的復制過程.

關于同步 IO 和異步 IO

根據 POSIX 定義:

  • A synchronous I/O operation causes the requesting process to be blocked until that I/O operation completes(導致請求進程阻塞, 直到 IO 操作完成).
  • An asynchronous I/O operation does not cause the requesting process to be blocked(不導致請求進程阻塞).
    根據上述定義, 我們的前四種模型: 阻塞 IO 模型, 非阻塞 IO 模型, IO 復用模型和信號驅動 IO 模型都是同步 IO 模型, 因為其中真正的 IO 操作(recvfrom 調用) 會阻塞進程(因為當有數據時, recvfrom 會阻塞等待內核將數據從內核空間復制到應用進程空間, 當賦值完成后, recvfrom 才返回.) 只有異步 IO 模型與 POSIX 定義的異步 IO 相匹配.

總結

在處理網絡 IO 操作時, 阻塞和非阻塞 IO 都是同步 IO.
只有調用了特殊的 API 才是異步 IO.

同步/異步 IO

因此網上常說的 "同步阻塞 IO", "同步非阻塞 IO" 其實就是阻塞 IO 模型和非阻塞 IO 模型, 因為阻塞 IO 和非阻塞 IO 模型都是同步的, 加了 "同步" 二字其實是多余了.
網絡上常說的 "異步非阻塞 IO" 其實就是異步 IO 模型.

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,923評論 6 535
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,740評論 3 420
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,856評論 0 380
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,175評論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,931評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,321評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,383評論 3 443
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,533評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,082評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,891評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,067評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,618評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,319評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,732評論 0 27
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,987評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,794評論 3 394
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,076評論 2 375

推薦閱讀更多精彩內容