淺談Linux 網(wǎng)絡(luò) I/O 模型簡(jiǎn)介(圖文)
1、介紹
Linux 的內(nèi)核將所有外部設(shè)備都看做一個(gè)文件來(lái)操作(一切皆文件),對(duì)一個(gè)文件的讀寫操作會(huì)調(diào)用內(nèi)核提供的系統(tǒng)命令,返回一個(gè)file descriptor(fd,文件描述符)。而對(duì)一個(gè)socket的讀寫也會(huì)有響應(yīng)的描述符,稱為socket fd(socket文件描述符),描述符就是一個(gè)數(shù)字,指向內(nèi)核中的一個(gè)結(jié)構(gòu)體(文件路徑,數(shù)據(jù)區(qū)等一些屬性)。
根據(jù)UNIX網(wǎng)絡(luò)編程對(duì)I/O模型的分類,UNIX提供了5種I/O模型。
1.1、阻塞I/O模型
最常用的I/O模型,默認(rèn)情況下,所有文件操作都是阻塞的。
比如I/O模型下的套接字接口:在進(jìn)程空間中調(diào)用recvfrom,其系統(tǒng)調(diào)用直到數(shù)據(jù)包到達(dá)且被復(fù)制到應(yīng)用進(jìn)程的緩沖區(qū)中或者發(fā)生錯(cuò)誤時(shí)才返回,在此期間一直等待。
進(jìn)程在調(diào)用recvfrom開始到它返回的整段時(shí)間內(nèi)都是被阻塞的,所以叫阻塞I/O模型。
圖示:

1.2、非阻塞I/O模型
recvfrom從應(yīng)用層到內(nèi)核的時(shí)候,就直接返回一個(gè)EWOULDBLOCK錯(cuò)誤,一般都對(duì)非阻塞I/O模型進(jìn)行輪詢檢查這個(gè)狀態(tài),看內(nèi)核是不是有數(shù)據(jù)到來(lái)。
圖示:

1.3、I/O復(fù)用模型
Linux提供select/poll,進(jìn)程通過(guò)將一個(gè)或多個(gè)fd傳遞給select或poll系統(tǒng)調(diào)用,阻塞在select操作上,這樣,select/poll可以幫我們偵測(cè)多個(gè)fd是否處于就緒狀態(tài)。
select/poll是順序掃描fd是否就緒,而且支持的fd數(shù)量有限,因此它的使用受到了一些制約。
Linux還提供一個(gè)epoll系統(tǒng)調(diào)用,epoll使用基于事件驅(qū)動(dòng)方式代替順序掃描,因此性能更高。當(dāng)有fd就緒時(shí),立即回調(diào)函數(shù)rollback。
圖示:

1.4、信號(hào)驅(qū)動(dòng)I/O模型
首先開啟套接口信號(hào)驅(qū)動(dòng)I/O功能,并通過(guò)系統(tǒng)調(diào)用sigaction執(zhí)行一個(gè)信號(hào)處理函數(shù)(此系統(tǒng)調(diào)用立即返回,進(jìn)程繼續(xù)工作,非阻塞)。當(dāng)數(shù)據(jù)準(zhǔn)備就緒時(shí),就為改進(jìn)程生成一個(gè)SIGIO信號(hào),通過(guò)信號(hào)回調(diào)通知應(yīng)用程序調(diào)用recvfrom來(lái)讀取數(shù)據(jù),并通知主循環(huán)函數(shù)處理樹立。
圖示:

1.5、異步I/O
告知內(nèi)核啟動(dòng)某個(gè)操作,并讓內(nèi)核在整個(gè)操作完成后(包括數(shù)據(jù)的復(fù)制)通知進(jìn)程。
信號(hào)驅(qū)動(dòng)I/O模型通知的是何時(shí)可以開始一個(gè)I/O操作,異步I/O模型有內(nèi)核通知I/O操作何時(shí)已經(jīng)完成。
圖示:

2、I/O多路復(fù)用技術(shù)
I/O編程中,需要處理多個(gè)客戶端接入請(qǐng)求時(shí),可以利用多線程或者I/O多路復(fù)用技術(shù)進(jìn)行處理。
正如前面的簡(jiǎn)介,I/O多路復(fù)用技術(shù)通過(guò)把多個(gè)I/O的阻塞復(fù)用到同一個(gè)select的阻塞上,從而使得系統(tǒng)在單線程的情況下可以同時(shí)處理多個(gè)客戶端請(qǐng)求。
與傳統(tǒng)的多線程模型相比,I/O多路復(fù)用的最大優(yōu)勢(shì)就是系統(tǒng)開銷小,系統(tǒng)不需要?jiǎng)?chuàng)建新的額外線程,也不需要維護(hù)這些線程的運(yùn)行,降低了系統(tǒng)的維護(hù)工作量,節(jié)省了系統(tǒng)資源。
主要的應(yīng)用場(chǎng)景:
- 服務(wù)器需要同時(shí)處理多個(gè)處于監(jiān)聽狀態(tài)或多個(gè)連接狀態(tài)的套接字。
- 服務(wù)器需要同時(shí)處理多種網(wǎng)絡(luò)協(xié)議的套接字。
支持I/O多路復(fù)用的系統(tǒng)調(diào)用主要有select、pselect、poll、epoll。
而當(dāng)前推薦使用的是epoll,優(yōu)勢(shì)如下:
- 支持一個(gè)進(jìn)程打開的socket fd不受限制。
- I/O效率不會(huì)隨著fd數(shù)目的增加而線性下將。
- 使用mmap加速內(nèi)核與用戶空間的消息傳遞。
- epoll擁有更加簡(jiǎn)單的API。
3、Java中的網(wǎng)絡(luò)IO編程
如果只是做Java開發(fā),以上內(nèi)容只需了解即可,不必深究(隨便說(shuō)說(shuō)而已)。
已專門出了文章介紹:Java 網(wǎng)絡(luò)IO編程總結(jié)(BIO、NIO、AIO均含完整實(shí)例代碼)
相關(guān)文章
Linux常用查看硬件設(shè)備信息命令大全(值得收藏)
本文是小編收藏整理的關(guān)于linux查看硬件設(shè)備信息的命名,非常不錯(cuò),值得收藏,需要的朋友參考下吧2016-12-12
Apache?Linkis?中間件架構(gòu)及快速安裝步驟
作為計(jì)算中間件,Linkis 提供了強(qiáng)大的連通、復(fù)用、編排、擴(kuò)展和治理管控能力,通過(guò)計(jì)算中間件將應(yīng)用層和引擎層解耦,簡(jiǎn)化了復(fù)雜的網(wǎng)絡(luò)調(diào)用關(guān)系,本文給大家介紹Apache?Linkis?中間件架構(gòu)的相關(guān)知識(shí),感興趣的朋友一起看看吧2022-03-03
如何在 Linux 中查找一個(gè)命令或進(jìn)程的執(zhí)行時(shí)間
在類 Unix 系統(tǒng)中,你可能知道一個(gè)命令或進(jìn)程開始執(zhí)行的時(shí)間,以及一個(gè)進(jìn)程運(yùn)行了多久。 但是,你如何知道這個(gè)命令或進(jìn)程何時(shí)結(jié)束或者它完成運(yùn)行所花費(fèi)的總時(shí)長(zhǎng)呢?下面小編給大家?guī)?lái)了在 Linux 中查找一個(gè)命令或進(jìn)程的執(zhí)行時(shí)間,一起看看吧2018-11-11
Ubuntu 20.04 CUDA&cuDNN安裝方法(圖文教程)
這篇文章主要介紹了Ubuntu 20.04 CUDA&cuDNN安裝方法(圖文教程),文中通過(guò)圖文代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07
解決Ubuntu 16.04下提示boot分區(qū)空間不足的辦法
最近看了看/boot的大小,發(fā)現(xiàn)幾次升級(jí)后,大小不足,所以想擴(kuò)容,一開始還想用磁盤操作,但上網(wǎng)查詢后發(fā)現(xiàn),磁盤操作實(shí)在風(fēng)險(xiǎn)太大,特別是雙系統(tǒng)的Linux,操作又是很麻煩,后來(lái)發(fā)現(xiàn)可以刪除多余的舊內(nèi)核來(lái)清理/boot,釋放空間。下面來(lái)看看詳細(xì)的解決方法吧。2017-02-02
xampp安裝后Apache無(wú)法啟動(dòng)解決辦法
XAMPP?(Apache+MySQL+PHP+PERL)是一個(gè)功能強(qiáng)大的建 XAMPP 軟件站集成軟件包,本文介紹了xampp安裝后Apache無(wú)法啟動(dòng)解決辦法,感興趣的可以了解一下2022-03-03
linux中的系統(tǒng)掛載(卸載)U盤(文件系統(tǒng))
這篇文章主要介紹了linux中的系統(tǒng)掛載(卸載)U盤(文件系統(tǒng)),具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-09-09

