轉載:http://blog.csdn.net/anxpp/article/details/51503329html
Linux 的內核將全部外部設備都看作一個文件來操做(一切皆文件),對一個文件的讀寫操做會調用內核提供的系統命令,返回一個file descriptor(fd,文件描述符)。而對一個socket的讀寫也會有響應的描述符,稱爲socket fd(socket文件描述符),描述符就是一個數字,指向內核中的一個結構體(文件路徑,數據區等一些屬性)。編程
根據UNIX網絡編程對I/O模型的分類,UNIX提供了5種I/O模型。服務器
最經常使用的I/O模型,默認狀況下,全部文件操做都是阻塞的。網絡
好比I/O模型下的套接字接口:在進程空間中調用recvfrom,其系統調用直到數據包到達且被複制到應用進程的緩衝區中或者發生錯誤時才返回,在此期間一直等待。多線程
進程在調用recvfrom開始到它返回的整段時間內都是被阻塞的,因此叫阻塞I/O模型。異步
圖示:socket
recvfrom從應用層到內核的時候,就直接返回一個EWOULDBLOCK錯誤,通常都對非阻塞I/O模型進行輪詢檢查這個狀態,看內核是否是有數據到來。函數
圖示:post
Linux提供select/poll,進程經過將一個或多個fd傳遞給select或poll系統調用,阻塞在select操做上,這樣,select/poll能夠幫咱們偵測多個fd是否處於就緒狀態。性能
select/poll是順序掃描fd是否就緒,並且支持的fd數量有限,所以它的使用受到了一些制約。
Linux還提供一個epoll系統調用,epoll使用基於事件驅動方式代替順序掃描,所以性能更高。當有fd就緒時,當即回調函數rollback。
圖示:
首先開啓套接口信號驅動I/O功能,並經過系統調用sigaction執行一個信號處理函數(此係統調用當即返回,進程繼續工做,非阻塞)。當數據準備就緒時,就爲改進程生成一個SIGIO信號,經過信號回調通知應用程序調用recvfrom來讀取數據,並通知主循環函數處理樹立。
圖示:
告知內核啓動某個操做,並讓內核在整個操做完成後(包括數據的複製)通知進程。
信號驅動I/O模型通知的是什麼時候能夠開始一個I/O操做,異步I/O模型有內核通知I/O操做什麼時候已經完成。
圖示:
I/O編程中,須要處理多個客戶端接入請求時,能夠利用多線程或者I/O多路複用技術進行處理。
正如前面的簡介,I/O多路複用技術經過把多個I/O的阻塞複用到同一個select的阻塞上,從而使得系統在單線程的狀況下能夠同時處理多個客戶端請求。
與傳統的多線程模型相比,I/O多路複用的最大優點就是系統開銷小,系統不須要建立新的額外線程,也不須要維護這些線程的運行,下降了系統的維護工做量,節省了系統資源。
主要的應用場景:
支持I/O多路複用的系統調用主要有select、pselect、poll、epoll。
而當前推薦使用的是epoll,優點以下:
若是隻是作Java開發,以上內容只需瞭解便可,沒必要深究(隨便說說而已)。
已專門出了文章介紹:Java 網絡IO編程總結(BIO、NIO、AIO均含完整實例代碼)