自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

詳解BIO與NIO的區(qū)別

開發(fā) 后端
IO讀寫時,多路復用機制都會依賴對一個事件多路分離器,負責把源事件的IO 事件分離出來,分別到相應的read/write事件分離器。涉及到事件分離器的兩種模式分別就是 Reactor和Proactor,Reactor是基于同步IO的,Proactor是基于異步IO的。

一、 Reactor and Proactor

IO讀寫時,多路復用機制都會依賴對一個事件多路分離器,負責把源事件的IO 事件分離出來,分別到相應的read/write事件分離器。涉及到事件分離器的兩種模式分別就是 Reactor和Proactor,Reactor是基于同步IO的,Proactor是基于異步IO的。

在Reactor模式中,事件分離者等待某個事件或者可應用或個操作的狀態(tài)發(fā)生(比如文件描述符可讀寫,或者是socket可讀寫),事件分離者就把這個事件傳給事先注冊的事件處理函數或者回調函數,由后者來做實際的讀寫操作。

在Proactor模式中,事件處理者(或者代由事件分離者發(fā)起)直接發(fā)起一個異步讀寫操作(相當于請求),而實際的工作是由操作系統(tǒng)來完成的。發(fā)起時,需要提供的參數包括用于存放讀到數據的緩存區(qū),讀的數據大小,或者用于存放外發(fā)數據的緩存區(qū),以及這個請求完后的回調函數等信息。事件分離者得知了這個請求,它默默等待這個請求的完成,然后轉發(fā)完成事件給相應的事件處理者或者回調。舉例來說,在Windows上事件處理者投遞了一個異步IO操作(稱有 overlapped的技術),事件分離者等IOCompletion事件完成. 這種異步模式的典型實現是基于操作系統(tǒng)底層異步API的,所以我們可稱之為“系統(tǒng)級別”的或者“真正意義上”的異步,因為具體的讀寫是由操作系統(tǒng)代勞的。

舉個例子,將有助于理解Reactor與Proactor二者的差異,以讀操作為例(類操作類似)。

在Reactor中實現讀:

- 注冊讀就緒事件和相應的事件處理器

- 事件分離器等待事件

- 事件到來,激活分離器,分離器調用事件對應的處理器。

- 事件處理器完成實際的讀操作,處理讀到的數據,注冊新的事件,然后返還控制權。

與如下Proactor(真異步)中的讀過程比較:

- 處理器發(fā)起異步讀操作(注意:操作系統(tǒng)必須支持異步IO)。在這種情況下,處理器無視IO就緒事件,它關注的是完成事件。

- 事件分離器等待操作完成事件

- 在分離器等待過程中,操作系統(tǒng)利用并行的內核線程執(zhí)行實際的讀操作,并將結果數據存入用戶自定義緩沖區(qū),最后通知事件分離器讀操作完成。

- 事件分離器呼喚處理器。

- 事件處理器處理用戶自定義緩沖區(qū)中的數據,然后啟動一個新的異步操作,并將控制權返回事件分離器。

可以看出,兩個模式的相同點,都是對某個IO事件的事件通知(即告訴某個模塊,這個IO操作可以進行或已經完成)。在結構

上,兩者也有相同點:demultiplexor負責提交IO操作(異步)、查詢設備是否可操作(同步),然后當條件滿足時,就回調handler;

不同點在于,異步情況下(Proactor),當回調handler時,表示IO操作已經完成;同步情況下(Reactor),回調handler時,表示

IO設備可以進行某個操作(can read or can write),handler這個時候開始提交操作。

二、BIO、NIO、AIO

NIO通常采用Reactor模式,AIO通常采用Proactor模式。AIO簡化了程序的編寫,stream的讀取和寫入都有OS來完成,不需要像NIO那樣子遍歷Selector。Windows基于IOCP實現AIO,Linux只有eppoll模擬實現了AIO。

Java7之前的JDK只支持NIO和BIO,從7開始支持AIO。

4種通信方式:TCP/IP+BIO, TCP/IP+NIO, UDP/IP+BIO, UDP/IP+NIO。

TCP/IP+BIO、

Socket和ServerSocket實現,ServerSocket實現Server端端口監(jiān)聽,Socket用于建立網絡IO連接。

不適用于處理多個請求 1.生成Socket會消耗過多的本地資源。2. Socket連接的建立一般比較慢。

BIO情況下,能支持的連接數有限,一般都采取accept獲取Socket以后采用一個thread來處理,one connection one thread。無論連接是否有真正數據請求,都需要獨占一個thread。

可以通過設立Socket池來一定程度上解決問題,但是使用池需要注意的問題是:1. 競爭等待比較多。 2. 需要控制好超時時間。

TCP/IP+NIO

使用Channel(SocketChannel和ServerSocketChannel)和Selector。

Server端通常由一個thread來監(jiān)聽connect事件,另外多個thread來監(jiān)聽讀寫事件。這樣做的好處是這些連接只有在真是請求的時候才會創(chuàng)建thread來處理,one request one thread。這種方式在server端需要支持大量連接但這些連接同時發(fā)送請求的峰值不會很多的時候十分有效。

UDP/IP+BIO

DatagramSocket和DatagramPacket。DatagramSocket負責監(jiān)聽端口以及讀寫數據,DatagramPacket作為數據流對象進行傳輸。

UDP/IP是無連接的,無法進行雙向通信,除非雙方都成為UDP Server。

UDP/IP+NIO

通過DatagramChannel和ByteBuffer實現。DatagramChannel負責端口監(jiān)聽及讀寫。ByteBuffer負責數據流傳輸。

如果要將消息發(fā)送到多臺機器,如果為每個目標機器都建立一個連接的話,會有很大的網絡流量壓力。這時候可以使用基于UDP/IP的Multicast協(xié)議傳輸,Java中可以通過MulticastSocket和DatagramPacket來實現。

Multicast一般多用于多臺機器的狀態(tài)同步,比如JGroups。SRM, URGCP都是Multicast的實現方式。eBay就采用SRM來實現將數據從主數據庫同步到各個搜索節(jié)點機器。

【編輯推薦】

  1. 探秘JDK 7之四:下一代I/O(NIO.2)
  2. 性能優(yōu)化總結:CPU和Load、NIO以及多線程
  3. Java 7 NIO2高性能Web服務器并發(fā)連接的處理
  4. 初試Java 7 NIO2:實現高性能的HTTP Server
  5. Google技術演講介紹Java 7 NIO.2概覽
責任編輯:金賀 來源: JavaEye博客
相關推薦

2018-09-19 14:53:02

NIOBIO運行

2020-04-16 15:20:43

PHP前端BIO

2023-07-11 08:40:02

IO模型后臺

2022-04-16 16:52:24

Netty網絡服務器客戶端程序

2023-06-26 07:39:10

2019-04-16 09:11:34

TomcatBIONIO

2023-03-07 08:00:12

netpollGo

2020-10-10 19:37:27

BIO 、NIO 、A

2024-11-06 16:38:51

IO網絡

2022-04-13 07:59:23

IOBIONIO

2021-06-11 17:26:06

代碼Java網絡編程

2023-03-31 07:49:51

syscall庫Echo Serve

2011-12-07 14:57:44

JavaNIO

2023-03-09 08:22:57

Go net庫Socket

2019-10-18 08:22:43

BIONIOAIO

2009-07-09 16:01:27

2013-03-21 15:27:09

LinuxUnix

2021-08-12 18:48:31

響應式編程Bio

2023-03-06 08:37:58

JavaNIO

2020-09-07 13:19:11

IO JavaBIO
點贊
收藏

51CTO技術棧公眾號