Java NIO系列教程(九) ServerSocketChannel

来源:互联网 发布:同花顺mac版功能全吗 编辑:程序博客网 时间:2024/05/16 02:45

本文转载至:http://ifeve.com/server-socket-channel/

Java NIO中的 ServerSocketChannel 是一个可以监听新进来的TCP连接的通道, 就像标准IO中的ServerSocket一样。ServerSocketChannel类在 java.nio.channels包中。


这里有个例子:

01ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
02 
03serverSocketChannel.socket().bind(new InetSocketAddress(9999));
04 
05while(true){
06    SocketChannel socketChannel =
07            serverSocketChannel.accept();
08 
09    //do something with socketChannel...
10}

打开 ServerSocketChannel

通过调用 ServerSocketChannel.open() 方法来打开ServerSocketChannel.如:

1ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();

关闭 ServerSocketChannel

通过调用ServerSocketChannel.close() 方法来关闭ServerSocketChannel. 如:

1serverSocketChannel.close();

监听新进来的连接

通过 ServerSocketChannel.accept() 方法监听新进来的连接。当 accept()方法返回的时候,它返回一个包含新进来的连接的 SocketChannel。因此, accept()方法会一直阻塞到有新连接到达。

通常不会仅仅只监听一个连接,在while循环中调用 accept()方法. 如下面的例子:

1while(true){
2    SocketChannel socketChannel =
3            serverSocketChannel.accept();
4 
5    //do something with socketChannel...
6}

当然,也可以在while循环中使用除了true以外的其它退出准则。

非阻塞模式

ServerSocketChannel可以设置成非阻塞模式。在非阻塞模式下,accept() 方法会立刻返回,如果还没有新进来的连接,返回的将是null。 因此,需要检查返回的SocketChannel是否是null.如:

查看源代码
打印帮助
01ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
02 
03serverSocketChannel.socket().bind(new InetSocketAddress(9999));
04serverSocketChannel.configureBlocking(false);
05 
06while(true){
07    SocketChannel socketChannel =
08            serverSocketChannel.accept();
09 
10    if(socketChannel != null){
11        //do something with socketChannel...
12    }
13}
java 示例代码:

package com.zzg.nio.server;import java.net.InetSocketAddress;import java.nio.ByteBuffer;import java.nio.channels.SelectionKey;import java.nio.channels.Selector;import java.nio.channels.ServerSocketChannel;import java.nio.channels.SocketChannel;import java.util.Iterator;public class NioServer {// 通道管理器private Selector selector;public void initServer(int port) throws Exception {// 获得一个ServerSocket通道ServerSocketChannel serverChannel = ServerSocketChannel.open();// 设置通道为 非阻塞serverChannel.configureBlocking(false);// 将该通道对于的serverSocket绑定到port端口serverChannel.socket().bind(new InetSocketAddress(port));// 获得一耳光通道管理器this.selector = Selector.open();// 将通道管理器和该通道绑定,并为该通道注册selectionKey.OP_ACCEPT事件// 注册该事件后,当事件到达的时候,selector.select()会返回,// 如果事件没有到达selector.select()会一直阻塞serverChannel.register(selector, SelectionKey.OP_ACCEPT);}// 采用轮训的方式监听selector上是否有需要处理的事件,如果有,进行处理public void listen() throws Exception {System.out.println("start server");// 轮询访问selectorwhile (true) {// 当注册事件到达时,方法返回,否则该方法会一直阻塞selector.select();// 获得selector中选中的相的迭代器,选中的相为注册的事件Iterator ite = this.selector.selectedKeys().iterator();while (ite.hasNext()) {SelectionKey key = (SelectionKey) ite.next();// 删除已选的key 以防重负处理ite.remove();// 客户端请求连接事件if (key.isAcceptable()) {ServerSocketChannel server = (ServerSocketChannel) key.channel();// 获得和客户端连接的通道SocketChannel channel = server.accept();// 设置成非阻塞channel.configureBlocking(false);// 在这里可以发送消息给客户端channel.write(ByteBuffer.wrap(new String("hello client").getBytes()));// 在客户端 连接成功之后,为了可以接收到客户端的信息,需要给通道设置读的权限channel.register(this.selector, SelectionKey.OP_READ);// 获得了可读的事件} else if (key.isReadable()) {read(key);}}}}// 处理 读取客户端发来的信息事件private void read(SelectionKey key) throws Exception {// 服务器可读消息,得到事件发生的socket通道SocketChannel channel = (SocketChannel) key.channel();// 穿件读取的缓冲区ByteBuffer buffer = ByteBuffer.allocate(18);channel.read(buffer);byte[] data = buffer.array();String msg = new String(data).trim();System.out.println("server receive from client: " + msg);ByteBuffer outBuffer = ByteBuffer.wrap(msg.getBytes());channel.write(outBuffer);}public static void main(String[] args) throws Throwable {  NioServer server = new NioServer();          server.initServer(8989);          server.listen();      }  }


0 0