netty源码推荐:https://www.cnblogs.com/crazymakercircle
附上个人理解的 JavaNio在netty 4.x中对应 代码位置
static class Server { public static void testServer() throws IOException { // 1、获取Selector选择器 Selector selector = Selector.open(); // 2、获取通道 【NioServerSocketChannel.newSocket】 ServerSocketChannel serverSocketChannel = ServerSocketChannel.open(); // 3.设置为非阻塞 【AbstractNioChannel 的构造方法里面】 serverSocketChannel.configureBlocking(false); // 4、绑定连接 【NioServerSocketChannel.doBind】 serverSocketChannel.bind(new InetSocketAddress(SystemConfig.SOCKET_SERVER_PORT)); // 5、将通道注册到选择器上,并注册的操作为:“接收”操作 【AbstractNioChannel.doRegister】 serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT); // 6、采用轮询的方式,查询获取“准备就绪”的注册过的操作 【NioEventLoop.select】 while (selector.select() > 0) { // 7、获取当前选择器中所有注册的选择键(“已经准备就绪的操作”) 【NioEventLoop.processSelectedKeysOptimized中的方法】 Iterator<SelectionKey> selectedKeys = selector.selectedKeys().iterator(); while (selectedKeys.hasNext()) { // 8、获取“准备就绪”的时间 SelectionKey selectedKey = selectedKeys.next(); // 下面的方法等价于 【NioEventLoop.processSelectedKey】 // 9、判断key是具体的什么事件 if (selectedKey.isAcceptable()) { // 10、若接受的事件是“接收就绪” 操作,就获取客户端连接 SocketChannel socketChannel = serverSocketChannel.accept(); // 11、切换为非阻塞模式 socketChannel.configureBlocking(false); // 12、将该通道注册到selector选择器上 socketChannel.register(selector, SelectionKey.OP_READ); } else if (selectedKey.isReadable()) { // 13、获取该选择器上的“读就绪”状态的通道 SocketChannel socketChannel = (SocketChannel) selectedKey.channel(); // 14、读取数据 ByteBuffer byteBuffer = ByteBuffer.allocate(1024); int length = 0; while ((length = socketChannel.read(byteBuffer)) != -1) { byteBuffer.flip(); System.out.println(new String(byteBuffer.array(), 0, length)); byteBuffer.clear(); } socketChannel.close(); } // 15、移除选择键 selectedKeys.remove(); } } // 7、关闭连接 serverSocketChannel.close(); } }
从ServerBootstrap的bind方法执行的顺序可以参考下图