首頁  >  文章  >  Java  >  JAVA-7NIO之Socket/ServerSocket通道

JAVA-7NIO之Socket/ServerSocket通道

巴扎黑
巴扎黑原創
2017-06-26 09:58:011001瀏覽

一、ServerSocketChannel

Java NIO中的 ServerSocketChannel 是一個可以監聽新進來的TCP連線的通道, 就像標準IO中的ServerSocket一樣。 ServerSocketChannel類別在 java.nio.channels套件中。

開啟ServerSocketChannel

透過呼叫ServerSocketChannel.open() 方法來開啟ServerSocketChannel.

關閉ServerSocketChannel

#透過呼叫ServerSocketChannel.close() 方法來關閉ServerSocketChannel. 

監聽新進來的連線

透過ServerSocketChannel.accept() 方法監聽新進來的連線。當 accept()方法返回的時候,它回傳一個包含新進來的連接的 SocketChannel。因此, accept()方法會一直阻塞到有新連接到達。

通常不會只監聽一個連接,在while循環中調用 accept()方法. 

當然,也可以在while循環中使用除了true以外的其它退出準則。

非阻塞模式

ServerSocketChannel可以設定成非阻塞模式。在非阻塞模式下,accept() 方法會立刻傳回,如果還沒有新進來的連接,返回的將是null。 因此,需要檢查返回的SocketChannel是否是null.如:

    /** * socket server channel     */@Testpublic void text2() throws IOException {
        ServerSocketChannel channel = ServerSocketChannel.open();    //新建channelchannel.socket().bind(new InetSocketAddress(9999));     //监听端口channel.configureBlocking(true);                             //设置阻塞while (true) {
            SocketChannel accept = channel.accept();                    //设置为阻塞,则此方法阻塞,直到有连接//如果设置为非阻塞,需要在这里判断 accept == null?ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
            accept.read(byteBuffer);
            byteBuffer.flip();                                    //反转while (byteBuffer.hasRemaining()) {                   //判断System.err.println((char)byteBuffer.get());       //输出            }
        }
    }

 

二、SocketChannel

Java NIO中的SocketChannel是一個連接到TCP網路套接字的通道。可以透過以下2種方式建立SocketChannel:

  1. 開啟一個SocketChannel並連接到網際網路上的某台伺服器。

  2. 一個新連線到達ServerSocketChannel時,會建立一個SocketChannel。

打開SocketChannel

下面是SocketChannel的開啟方式:

關閉SocketChannel

當用完SocketChannel之後呼叫SocketChannel.close ()關閉SocketChannel:

從SocketChannel 讀取資料

要從SocketChannel讀取數據,呼叫一個read()的方法之一。

首先,先分配一個Buffer。從SocketChannel讀取到的資料將會放到這個Buffer中。

然後,呼叫SocketChannel.read()。該方法將資料從SocketChannel 讀取到Buffer中。 read()方法傳回的int值表示讀了多少位元組進Buffer裡。如果傳回的是-1,表示已經讀到了流的末尾(連接關閉了)。

寫入 SocketChannel

寫資料到SocketChannel用的是SocketChannel.write()方法,該方法以一個Buffer作為參數。

注意SocketChannel.write()方法的呼叫是在一個while循環中的。 Write()方法無法保證能寫多少位元組到SocketChannel。所以,我們重複呼叫write()直到Buffer沒有要寫的位元組為止。

非阻塞模式

可以設定SocketChannel 為非阻塞模式(non-blocking mode).設定之後,就可以在非同步模式下呼叫connect(), read() 和write()了。

connect()

如果SocketChannel在非阻塞模式下,此時呼叫connect(),則該方法可能在連線建立之前就回傳了。為了確定連線是否建立,可以呼叫finishConnect()的方法。

 

write()

非阻塞模式下,write()方法在尚未寫出任何內容時可能就回傳了。所以需要在循環中調用write()。前面已經有例子了,這裡就不贅述了。

read()

非阻塞模式下,read()方法在尚未讀取到任何資料時可能就回傳了。所以需要注意它的int回傳值,它會告訴你讀了多少位元組。

非阻塞模式與選擇器

非阻塞模式與選擇器搭配會運作的更好,透過將一或多個SocketChannel註冊到Selector,可以詢問選擇器哪個通道已經準備好了讀取,寫入等。 Selector與SocketChannel的搭配使用會在後面詳講。

/** * socket channel     */@Testpublic void test3() throws IOException {
        SocketChannel channel = SocketChannel.open();                               //新建服务端channel.connect(new InetSocketAddress("127.0.0.1",9999));   //连接服务端地址ByteBuffer byteBuffer = ByteBuffer.allocate(1024);  //缓冲区byteBuffer.put("123".getBytes());
        byteBuffer.flip();                                  //反转while (byteBuffer.hasRemaining()) {                 //判断            channel.write(byteBuffer);
        }
    }

 

以上是JAVA-7NIO之Socket/ServerSocket通道的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn