关于 Java NIO 的 Selector 的事儿,这篇文章里面全都有

news2025/1/23 10:33:25

前面 4 篇文章深入分析了 NIO 三大组件中的两个:Buffer 和 Channel:

  • 【死磕 NIO】— 深入分析Buffer
  • 【死磕 NIO】— 深入分析Channel和FileChannel
  • 【死磕NIO】— 跨进程文件锁:FileLock
  • 【死磕NIO】— 探索 SocketChannel 的核心原理

这篇文章则介绍第三个组件:Selector。

相比 Buffer 和 Channel 而言,Selector 对于 NIO 来说显得更加重要,因为它是 NIO 实现多路复用的核心,它的使命就是完成 IO 的多路复用。

Selector 简介

在前一篇文章:【死磕 NIO】— ServerSocketChannel 的应用实例 ,大明哥分析了 ServerSocketChannel 两种模式的缺点

  • 阻塞模式:所有阻塞方法都会引起线程的暂停,根本无法应用到业务中来
  • 非阻塞模式:CPU 一直在空转,浪费资源

所以,如果是我们服务端单独使用 ServerSocketChannel 确实是很麻烦,典型的吃力不讨好。故而我们希望有一个组件能够统一管理我们的 Channel,这个组件就是选择器 Selector。

Selector(选择器)是 Channel 的多路复用器,它可以同时监控多个 Channel 的 IO 状况,允许单个线程来操作多个 Channel。如下:

Selector 的作用是什么?

Selector 提供选择执行已经就绪的任务的能力。从底层来看,Selector 提供了询问 Channel 是否已经准备好执行每个 I/O 操作的能力。Selector 允许单线程处理多个 Channel。仅用单个线程来处理多个 Channels 的好处是,只需要更少的线程来处理 Channel 。事实上,可以只用一个线程处理所有的通道,这样会大量的减少线程之间上下文切换的开销。

Selector 的使用

使用 Selector 的主要流程如下:

  1. 打开 Selector
  2. 将 Channel 注册到 Selector 中,并设置要监听的事件
  3. 轮询处理 IO 操作

打开 Selector

和 SocketChannel 相似,调用 Selector.open() 就可以打开一个选择器实例。

Selector selector = Selector.open();

注册 Selector

为了将 Channel 和 Selector 配合使用,我们需要将 Channel 注册到对应的 Selector 上,调用 SelectableChannel.register() 方法来实现。

channel.configureBlocking(false);
SelectionKey key = channel.register(selector,Selectionkey.OP_ACCEPT);

这里有一个要注意的地方,所有注册到 Selector 中的 Channel 都必须是非阻塞的。怎么判断 Channel 是否可以设置为非阻塞呢?判断它是否继承了SelectableChannel,SelectableChannel 是一个抽象类,它提供了实现 Channel 的可选择性所需要的公共方法。而 FileChannel 没有继承 SelectableChannel ,所以它不能使用 Selector。

register() 提供了两个参数,一个是要注册的 Selector 是谁,第二个参数是对什么事件感兴趣。事件类型有四种:

  • 连接 : SelectionKey.OP_CONNECT
  • 接收 : SelectionKey.OP_ACCEPT
  • 可读 : SelectionKey.OP_READ
  • 可写 : SelectionKey.OP_WRITE

如果感兴趣的事件不止一个,则可以使用“位运算 | ” 来组合多个事件,如: SelectionKey.OP_CONNECT | SelectionKey.OP_ACCEPT

需要提醒的是,Selector 关注的不是 Channel 的操作,而是 Channel的某个操作的一种就绪状态。一旦 Channel 具备完成某个操作的条件,表示该 Channel 的某个操作已经就绪,就可以被 Selector 查询到,程序可以对该 Channel 进行对应的操作。比如说,某个 SocketChannel 可以连接到一个服务器,则处于“连接就绪”(OP_CONNECT)。某给 ServerSocketChannel 可以接收新的连接,则处理“接收就绪”(SelectionKey.OP_ACCEPT)。

轮询处理 IO 操作

将 Channel 注册到 Selector 并关注相对应的时间后,就可以轮询处理 IO 事件了。

Selector 提供了方法 select(),该方法可以查询出已经就绪的 Channel操作,如果没有事件发生,则该方法会一直阻塞,直到有事件。select() 有三个重载方法:

  • select():阻塞到至少有一个通道在你注册的事件上就绪了。·
  • select(long timeout):和select()一样,但最长阻塞事件为timeout毫秒。
  • selectNow():非阻塞,只要有通道就绪就立刻返回。

select() 返回值为 int 类型,该值表示有多少 Channel 的操作已经就绪,更准确地说是上一次 select() 到这一次 select() 方法之间的时间段内,有多少 Channel 变成了就绪状态。

select() 返回后,如果返回值大于 0 ,则可以调用 selectedKeys() 方法,该方法返回一个 Set 集合,该集合是一个 SelectionKey 的集合,SelectionKey 表示的是可选择通道 SelectableChannel 和一个特定的 Selector之间的注册关系。

  • SelectionKey 是一个抽象类,表示 SelectableChannel 在 Selector 中注册的标识.每个 Channel 向 Selector 注册时,都将会创建一个selectionKey
  • SelectionKey 是 SelectableChannel 与 Selector 的建立关系,并维护了 Channel 事件
  • 可以通过 cancel() 方法取消 key,取消的 key 不会立即从 Selector 中移除,而是添加到 cancelledKeys 中,在下一次 select() 操作时移除它.所以在调用某个 key 时,需要使用 isValid 进行校验。

SelectionKey 提供了两个非常重要的 “Set”:interest setready set

  • interest set 表示感兴趣的事件,我们可以通过以下方式获取:
int interestSet = selectionKey.interestOps();

boolean isInterestedInAccept  = interestSet & SelectionKey.OP_ACCEPT;
boolean isInterestedInConnect = interestSet & SelectionKey.OP_CONNECT;
boolean isInterestedInRead    = interestSet & SelectionKey.OP_READ;
boolean isInterestedInWrite   = interestSet & SelectionKey.OP_WRITE;
  • ready set:代表了 Channel 所准备好了的操作。
int readySet = selectionKey.readyOps();
selectionKey.isAcceptable();
selectionKey.isConnectable();
selectionKey.isReadable();
selectionKey.isWritable();

以下代码是一个处理 IO 操作的完整代码:

while (true) {
    selector.select();
    
    Set<SelectionKey> selectedKeys = selector.selectedKeys();

    Iterator<SelectionKey> keyIterator = selectedKeys.iterator();
    while (iterator.hasNext()) {
        SelectionKey key = keyIterator.next();
    
        if(key.isAcceptable()) {
            // a connection was accepted by a ServerSocketChannel.
    
        } else if (key.isConnectable()) {
            // a connection was established with a remote server.
    
        } else if (key.isReadable()) {
            // a channel is ready for reading
    
        } else if (key.isWritable()) {
            // a channel is ready for writing
        }
        
        // 这段代码非常重要,后面演示
        key.remove();
    }
}

这里有一段非常重要的代码 key.remove(),这行代码表示,我已经在处理该 IO 事件了,需要删除。

实例

简单实例

下面大明哥用 Selector 实现一个完整的案例。

public static void main(String[] args) throws Exception {
    // 创建 ServerSocketChannel
    ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();

    // 设置为非阻塞
    serverSocketChannel.configureBlocking(false);

    // 绑定 8081 端口
    serverSocketChannel.bind(new InetSocketAddress(8081));

    // 打开 Selector
    Selector selector = Selector.open();

    // 将 SocketChannel 注册到  Selector
    // 通常我们都是先注册一个 OP_ACCEPT 事件, 然后在 OP_ACCEPT 到来时, 再将这个 Channel 的 OP_READ
    serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);

    while (true) {
        // select 方法,一直阻塞直到有事件发生
        selector.select();

        // 获取 I/O 操作就绪的 SelectionKey, 通过 SelectionKey 可以知道哪些 Channel 的哪类 I/O 操作已经就绪
        Iterator<SelectionKey> iterator = selector.selectedKeys().iterator();

        while (iterator.hasNext()) {
            SelectionKey key = iterator.next();

            // 获取一个 SelectionKey 后,我们要将其删除掉,表示我们已经处理了这个事件
            iterator.remove();

            if (key.isAcceptable()) {
                // 连接时间发生
                // 当客户端连接服务端的时候,我们需要服务单与之建立连接
                // 需要注意的是在 OP_ACCEPT 事件中, 从 key.channel() 返回的 Channel 是 ServerSocketChannel
                ServerSocketChannel socketChannel = (ServerSocketChannel) key.channel();
                // 需要从 socketChannel 获取 SocketChanel
                SocketChannel clientChannel = socketChannel.accept();
                log.info("{} 建立连接",clientChannel);
                // 设置 clientChannel 为非阻塞
                clientChannel.configureBlocking(false);

                clientChannel.register(selector,SelectionKey.OP_READ);
            } else if (key.isReadable()) {
                // 获取的为 SocketChannel
                SocketChannel clientChannel = (SocketChannel) key.channel();
                ByteBuffer byteBuffer = ByteBuffer.allocate(64);
                int size = clientChannel.read(byteBuffer);
                if (size < 0) {
                    // 小于 0 表示客户端断开连接,需要关闭该 SocketChannel
                    log.info("{},断开了连接",clientChannel);
                    clientChannel.close();
                } else {
                    byteBuffer.flip();

                    CharBuffer charBuffer = Charset.forName("utf-8").decode(byteBuffer);

                    log.info("{},发来了消息,消息内容是:{}",clientChannel,charBuffer.toString());

                    // 服务端接收消息后,给客户端发送给客户端
                    Scanner scanner = new Scanner(System.in);
                    String string = scanner.nextLine();
                    ByteBuffer writeBuffer = Charset.forName("utf-8").encode(string);
                    clientChannel.write(writeBuffer);

                    if (writeBuffer.hasRemaining()) {
                        // 如果不能一次性发完只需要触发 write 事件去发
                        key.attach(writeBuffer);
                        key.interestOps(key.interestOps() + SelectionKey.OP_WRITE);
                    }
                }
            } else if (key.isWritable() && key.isValid()) {
                ByteBuffer byteBuffer = (ByteBuffer) key.attachment();
                SocketChannel clientChannel = (SocketChannel) key.channel();
                byteBuffer.flip();

                clientChannel.write(byteBuffer);

                if (!byteBuffer.hasRemaining()) {
                    // 如果已完,则只无须关注 write 事件
                    key.attach(null);
                    key.interestOps(key.interestOps() - SelectionKey.OP_WRITE);
                }
            }
         }
    }
}

启动服务端,打开 iTerm,输入命令 telnet localhost 8081,连接服务端,这时服务端接收到客户端 client-01 的连接请求,进行建立连接。

建立连接后,客户端发送消息i am client_01,服务端收到消息,然后给客户端发送消息hi,client-01,i am server

  • 服务端

  • 客户端

分析为什么要:key.remove()

这里拿上面那个问题来说明,为什么要加这 key.remove() 代码呢?首先这段代码的意思是说获取一个 SelectionKey 后,我们需要将其删除,表示我们已经对该 IO 事件进行了处理,如果没有这样代码会有什么后果呢?报 NullPointerException

注释掉 key.remove() 这行代码,然后加一些日志,然后去掉服务端发送消息的代码,如下:

while (true) {
    // select 方法,一直阻塞直到有事件发生
    selector.select();

    // 获取 I/O 操作就绪的 SelectionKey, 通过 SelectionKey 可以知道哪些 Channel 的哪类 I/O 操作已经就绪
    Iterator<SelectionKey> iterator = selector.selectedKeys().iterator();

    while (iterator.hasNext()) {
        SelectionKey key = iterator.next();
        log.info("key={}", key);

        // 获取一个 SelectionKey 后,我们要将其删除掉,表示我们已经处理了这个事件
        //iterator.remove();

        if (key.isAcceptable()) {
            // 连接时间发生
            // 当客户端连接服务端的时候,我们需要服务单与之建立连接
            // 需要注意的是在 OP_ACCEPT 事件中, 从 key.channel() 返回的 Channel 是 ServerSocketChannel
            ServerSocketChannel socketChannel = (ServerSocketChannel) key.channel();
            // 需要从 socketChannel 获取 SocketChanel
            SocketChannel clientChannel = socketChannel.accept();
            log.info("{} 建立连接", clientChannel);
            // 设置 clientChannel 为非阻塞
            clientChannel.configureBlocking(false);

            clientChannel.register(selector, SelectionKey.OP_READ);
        } else if (key.isReadable()) {
            // 获取的为 SocketChannel
            SocketChannel clientChannel = (SocketChannel) key.channel();
            ByteBuffer byteBuffer = ByteBuffer.allocate(64);
            int size = clientChannel.read(byteBuffer);
            if (size < 0) {
                // 小于 0 表示客户端断开连接,需要关闭该 SocketChannel
                log.info("{},断开了连接", clientChannel);
                clientChannel.close();
            } else {
                byteBuffer.flip();

                CharBuffer charBuffer = Charset.forName("utf-8").decode(byteBuffer);

                log.info("{},发来了消息,消息内容是:{}", clientChannel, charBuffer.toString());
            }
        }
    }

    System.out.println("==============================我是分割线===================================");
}

启动服务端,然后客户端连接,发送消息,结果如下:

为什么会这样呢?这里我们来梳理整个流程。

  • 首先服务端创建一个 Selector,该 Selector 与 ServerSocketChannel 绑定,且关注 accept 事件。如下

  • 当客户端发起连接时,selector.selectedKeys() 会返回 Set 集合,该集合包含了已经准备就绪的 SelectionKey,这个时候只有连接事件,相对应的 SelectionKey 为 2b71fc7e

  • 当服务端与客户端建立连接后,绑定 Selector 并关注 read 事件。这里需要注意的是 Selector 并不会主动去删除 SelectionKey,它只会增加,所以这个时候 Selector 里面有两个 SelectionKey,一个是 2b71fc7e(accept),一个是 1de0aca6(read)。建立连接后,事件处理完成,会该事件与之对应的事件去掉,也就是 2b71fc7e 的 SelectionKey 绑定的 ServerSocketChannel ,但是 Selector 里面对应的 SelectionKey 还是存在的。

  • 当客户端给服务端发送消息时,服务端监测到有事件发生,会将发生时间的 SelectionKey@1de0aca6 加入到 selectedKey 中,如下:

在迭代过程第一次取的是 SelectionKey@1de0aca6,这个是读事件,可以正常读,打印客户端发送过来的,但是第二次读取的是 SelectionKey@2b71fc7e,但是这个 Key 与之相绑定的事件已经处理了,它为 null,那肯定会报 NullPointerException。所以在使用 NIO 时一定要主动删除已经处理过的 SelectionKey ,既主动调用 key.remove(),删除该 SelectionKey。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1209925.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ffmpeg5及以上-s和像素格式转换 画屏问题

环境: lsb_release -a No LSB modules are available. Distributor ID: Ubuntu Description: Ubuntu 22.10 Release: 22.10 Codename: kinetic拉下ffmpeg源码&#xff0c;6.0.1&#xff0c;4.3.6&#xff0c;5.1.4&#xff0c;依次安装作实验 ./configure --disable-x86asm …

msvcp140.dll丢失的解决方法、详细解析dll缺失原因及对电脑的影响

msvcp140.dll是一款Visual C Redistributable for Visual Studio 2015的运行时库&#xff0c;许多程序都需要依赖这个库才能正常运行。当msvcp140.dll丢失时&#xff0c;我们可能会遇到无法打开程序或游戏&#xff0c;甚至系统崩溃的问题。本文将详细介绍msvcp140.dll丢失的解决…

Linux--makefile

一、makefile的作用 makefile是一个文件&#xff0c;是围绕依赖关系和依赖方法的自动化编译工具 一个工程中的源文件有很多&#xff0c;按照不同的类型、功能、模块放在不同的目录中。而makefile定义了一系列的规则来指定&#xff0c;那些文件需要先编译&#xff0c;那些文件…

后端接口性能优化分析-程序结构优化

&#x1f44f;作者简介&#xff1a;大家好&#xff0c;我是爱吃芝士的土豆倪&#xff0c;24届校招生Java选手&#xff0c;很高兴认识大家&#x1f4d5;系列专栏&#xff1a;Spring源码、JUC源码&#x1f525;如果感觉博主的文章还不错的话&#xff0c;请&#x1f44d;三连支持&…

【开源】基于JAVA的电子元器件管理系统

目录 一、摘要1.1 项目简介1.2 项目详细录屏 二、研究内容三、界面展示3.1 登录&注册&主页3.2 元器件单位模块3.3 元器件仓库模块3.4 元器件供应商模块3.5 元器件品类模块3.6 元器件明细模块3.7 元器件类型模块3.8 元器件采购模块3.9 元器件领用模块3.10 系统基础模块 …

PlantUML基础使用教程

环境搭建 IDEA插件下载 打开IEDA系列IDE&#xff0c;从FIle–>Settings–>Plugins–>Marketplace 进入到插件下载界面&#xff0c;搜索PlantUML&#xff0c;安装PlantUML Integration和PlantUML Parser两个插件&#xff0c;并重启IDE 安装和配置Graphviz 进入官网…

【Python 千题 —— 基础篇】欢迎光临

题目描述 题目描述 欢迎光临。为列表中的每个嘉宾打印欢迎光临语句。例如&#xff0c;有一份嘉宾列表 ["李二狗", "王子鸣"]&#xff0c;则需要根据嘉宾名单打印输出&#xff1a; 欢迎光临&#xff01;李二狗。 欢迎光临&#xff01;王子鸣。下面是一份…

基于布谷鸟算法优化概率神经网络PNN的分类预测 - 附代码

基于布谷鸟算法优化概率神经网络PNN的分类预测 - 附代码 文章目录 基于布谷鸟算法优化概率神经网络PNN的分类预测 - 附代码1.PNN网络概述2.变压器故障诊街系统相关背景2.1 模型建立 3.基于布谷鸟优化的PNN网络5.测试结果6.参考文献7.Matlab代码 摘要&#xff1a;针对PNN神经网络…

编程怎么学习视频教程,编程实例入门教程,中文编程开发语言工具下载

编程怎么学习视频教程&#xff0c;编程实例入门教程&#xff0c;中文编程开发语言工具下载。 给大家分享一款中文编程工具&#xff0c;零基础轻松学编程&#xff0c;不需英语基础&#xff0c;编程工具可下载。 这款工具不但可以连接部分硬件&#xff0c;而且可以开发大型的软件…

csapp第三章读书笔记

caspp chapter 3 寄存器 operand form data movement instructions mov 指令例子: 0扩展 movz 指令: Zero-extending data movement instructions是一种计算机指令类型&#xff0c;涉及将数据从一个位置移动到另一个位置&#xff0c;同时通过在最重要的一端添加零位来将数据扩…

【考研复习】二叉树的特殊存储|三叉链表存储二叉树、一维数组存储二叉树、线索二叉树

文章目录 三叉链表存储二叉树三叉链表的前序遍历&#xff08;不使用栈&#xff09;法一三叉链表的前序遍历&#xff08;不使用栈&#xff09;法二 一维数组存储二叉树一维数组存储二叉树的先序遍历 线索二叉树的建立真题演练 三叉链表存储二叉树 三叉链表结构体表示如下图所示…

探秘Vue组件间通信:详解各种方式助你实现目标轻松搞定!

&#x1f3ac; 江城开朗的豌豆&#xff1a;个人主页 &#x1f525; 个人专栏 :《 VUE 》 《 javaScript 》 &#x1f4dd; 个人网站 :《 江城开朗的豌豆&#x1fadb; 》 ⛺️ 生活的理想&#xff0c;就是为了理想的生活 ! ​ 目录 ⭐ 专栏简介 &#x1f4d8; 文章引言 一…

软件工程理论与实践 (吕云翔) 第四章 结构化分析课后习题及答案

第四章 结构化分析 知识点&#xff1a; ​ 结构化分析模型的核心为数据字典&#xff0c;它是描述软件使用和产生的所有数据对象。围绕着这个核心有3种不同的图&#xff1a;“数据流图”指出当数据在软件系统中移动时怎样被变换&#xff0c;并描绘变换数据流的功能和子功能&am…

实现Vue3 readonly,教你如何一步步重构

本文通过实现readonly方法&#xff0c;一步步展示重构的流程。 前言 readonly接受一个对象&#xff0c;返回一个原值的只读代理。 实现 Vue3 中readonly方法&#xff0c;先来看一下它的使用。 <script setup> import { readonly } from "vue";let user {n…

Vue中methods实现原理

目录 前言 回调函数中的this指向问题 vue实例访问methods methods实现原理 前言 vue实例对象为什么可以访问methods中的函数方法&#xff1f;methods的实现原理是什么&#xff1f; 回调函数中的this指向问题 在解答前言中的问题前&#xff0c;需要了解一下回调函数中的th…

计算机 - - - 浏览器网页打开本地exe程序,网页打开微信,网页打开迅雷

效果 在电脑中安装了微信和迅雷&#xff0c;可以通过在地址栏中输入weixin:打开微信&#xff0c;输入magnet:打开迅雷。 同理&#xff1a;在网页中使用a标签&#xff0c;点击后跳转链接打开weixin:&#xff0c;也会同样打开微信。 运用同样的原理&#xff0c;在网页中点击超…

为什么PDF文件不能打印?

正常的PDF文件是可以打印的&#xff0c;如果PDF文件打开之后发现文件不能打印&#xff0c;我们需要先查看一下自己的打印机是否能够正常运行&#xff0c;如果打印机是正常的&#xff0c;我们再查看一下&#xff0c;文件中的打印功能按钮是否是灰色的状态。 如果PDF中的大多数功…

[工业自动化-20]:西门子S7-15xxx编程 - 软件编程 - 基本编程指令与梯形图基本元素:位逻辑指令、定时器指令、计数器指令、触发器指令

目录 一、PLC编程的基本指令 1.1 什么是PLC指令 1.2 PLC指令的分类 1.3 PLC指令与梯形图基本元素的关系 三、基本的位运算指令 四、边沿触发指令 4.1 什么是沿 4.2 沿的持续时间 4.3 使用场景 五、定时器指令 六、计数器指令 七、触发器指令 一、PLC编程的基本指令…

python语言的由来与发展历程

Python语言的由来可以追溯到1989年&#xff0c;由Guido van Rossum&#xff08;吉多范罗苏姆&#xff09;创造。在他的业余时间里&#xff0c;Guido van Rossum为了打发时间&#xff0c;决定创造一种新的编程语言。他受到了ABC语言的启发&#xff0c;ABC语言是一种过程式编程语…

DAY54 392.判断子序列 + 115.不同的子序列

392.判断子序列 题目要求&#xff1a;给定字符串 s 和 t &#xff0c;判断 s 是否为 t 的子序列。 字符串的一个子序列是原始字符串删除一些&#xff08;也可以不删除&#xff09;字符而不改变剩余字符相对位置形成的新字符串。&#xff08;例如&#xff0c;"ace"是…