NIO引入多路复用器实例

NIO引入多路复用器实例
之前的NIO实例是为了本文做准备的,java底层即使用一个Selector(多路复用器)轮询来实现NIO,其基本的思路与前面的例子相同。
NIO 有三大核心组件: Channel(通道), Buffer(缓冲区),Selector(多路复用器)
1、channel 类似于流,每个 channel 对应一个 buffer缓冲区,buffer 底层就是个数组
2、channel 会注册到 selector 上,由 selector 根据 channel 读写事件的发生将其交由某个空闲的线程处理
3、NIO 的 Buffer 和 channel 都是既可以读也可以写

NIO引入多路复用器代码示例:

public class NioServer {

    public static void main(String[] args) throws IOException {
        //1、开启多路复用器
        Selector selector = Selector.open();
        //2、服务端socket
        ServerSocketChannel socketChannel = ServerSocketChannel.open();
        //3、绑定端口
        socketChannel.bind(new InetSocketAddress(8897));
        //4、设置非阻塞
        socketChannel.configureBlocking(false);
        //5、注册并设置监听的事件
        socketChannel.register(selector, SelectionKey.OP_ACCEPT);
        System.out.println("服务器启动成功");

        //死循环
        for (;;) {
            System.out.println("此时没有客户端连接,select方法会阻塞轮询");
            //阻塞,当selector中有事件发生时会解阻塞
            selector.select();
            Set<SelectionKey> selectionKeys = selector.selectedKeys();
            Iterator<SelectionKey> iterator = selectionKeys.iterator();
            while (iterator.hasNext()) {
                SelectionKey key = iterator.next();
                //接受连接事件
                if (key.isAcceptable()) {
                    ServerSocketChannel channel = (ServerSocketChannel) key.channel();
                    //接收客户端连接
                    SocketChannel clientSocket = channel.accept();
                    //客户端读事件非阻塞
                    clientSocket.configureBlocking(false);
                    clientSocket.register(selector, SelectionKey.OP_READ);
                    System.out.println("客户端连接成功");
                }
                //读事件
                if (key.isReadable()) {
                    ByteBuffer buffer = ByteBuffer.allocate(128);
                    SocketChannel socketChannel1 = (SocketChannel) key.channel();
                    int len = socketChannel1.read(buffer);
                    if (len > 0) {
                        System.out.println("客户端传过来的数据是:" + new String(buffer.array()));
                    }
                    else if (len == -1) {
                        System.out.println("客户端未发送");
                    }
                }
                iterator.remove();
            }
        }
    }
}

客户端:

public class NioClient {

    public static void main(String[] args) throws IOException {
        Socket socket = new Socket();
        socket.connect(new InetSocketAddress("127.0.0.1", 8897));
        while (true) {
            Scanner sc = new Scanner(System.in);
            String msg = sc.next();
            socket.getOutputStream().write(msg.getBytes());
        }
    }
}

当启动多个客户端时,服务端控制台结果:
没有客户端连接:
在这里插入图片描述
客户端1,连接未发送数据:
在这里插入图片描述
客户端发送数据:
在这里插入图片描述
可以多个客户端连接测试。
总结:同步非阻塞,服务器实现模式为一个线程可以处理多个请求(连接),客户端发送的连接请求都会注册到多路复用器selector上,多路复用 器轮询到连接有IO请求就进行处理。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了安卓应用、小程序、Python应用和Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值