Java非阻塞I/O模型之NIO说明

news/2024/7/5 6:26:01
在了解NIO (Non-Block I/O) 非阻塞I/O模型之前,我们可以先了解一下原始的BIO(Block I/O) 阻塞I/O模型,NIO模型能够以非阻塞的方式更好的利用服务器资源,需要的朋友可以参考下

组件说明

(1)Channel:NIO模型中的管道,管道是链接建立和通信的重要组件,我们可以理解管道是一个容器环境,我们所有的I/O的建立读取都可以在这个容器中进行

(2)Selector:NIO中的选择器,NIO是由事件驱动的,当有链接事件或者读取事件发生时,这个事件可以注册到这个选择器上,并且最终被我们检测到。

(3)SelectionKey:我们可以在Selector中进行检测是否有SelectionKey产生,并且根据这个SelectionKey中的信息判断时什么事件发生了。

代码说明

(1)开启ServerSocketChannel,并开始监听

//初始化一个网络地址,并绑定7000端口号
InetSocketAddress inetSocketAddress = new InetSocketAddress(7000);
//ServerSocketChannel.open() 方法实例化一个ServerSocketChannel对象
ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
//serverSocketChannel绑定初始化的网络地址,并开始监听
serverSocketChannel.socket().bind(inetSocketAddress);
//将这个通道设置为非阻塞的
serverSocketChannel.configureBlocking(false);

(2)初始化选择器,并将这个选择器注册到上面的网络通道中

//得到一个Selector对象
Selector selector = Selector.open();
//在channel上注册selector,并且告诉这个选择器初始应该监听的事件,
//SelectionKey.OP_ACCEPT 为监听链接进入的事件,初始化并不监听数据读取的事件
//SelectionKey.OP_READ 事件读取事件,需要在有链接进入时,配合链接一起注册
serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);

(3)主循环

//循环等待客户端链接
while(true){//等待1秒,1秒内没有链接事件发生,直接返回if(selector.select(1000)==0){System.out.println("服务器等待了1秒,无连接进入");continue;}//有事件发生,拿到集合//selector.selectedKeys() 关注事件的集合//通过这个可以反向获取通道Set selectionKeys = selector.selectedKeys();//遍历集合,使用迭代器Iterator keyIterator = selectionKeys.iterator();while(keyIterator.hasNext()){//获取事件keySelectionKey key = keyIterator.next();//根据key对应的通道发生的事件做相应的处理if(key.isAcceptable()){//如果是ACCEPT事件,客户端链接//传统的accept()是阻塞的,但是在NIO中,当key.isAcceptable()方法返回true的时候,这个链接就已经存在了,所以accept()会立刻执行SocketChannel socketChannel = serverSocketChannel.accept();socketChannel.configureBlocking(false);//将当前的socketChannel注册的selector,关注事件为READ,同时给Channel关联一个BufferSelectionKey register = socketChannel.register(selector, SelectionKey.OP_READ, ByteBuffer.allocate(128));}else if(key.isReadable()){//发生了READ事件//通过key反向获取ChannelSocketChannel channel = (SocketChannel)key.channel();//获取到该channel关联的bufferByteBuffer buffer =(ByteBuffer) key.attachment();channel.read(buffer);System.out.println("From 客户端 :"+new String(buffer.array()));}//手动在集合中移除当前的SelectionKey否则可能会出现重复操作keyIterator.remove();}}

总结

(1)使用一个事件驱动的方式,在没有事件发生的时候,服务器可以去做一些自己需要做的事。

(2)当有事件发生的时候,通过Selector去关心是什么事件。

(3)甚至不需要使用多线程,就能同时处理更多的链接请求。

(4)当然我们也可以配合多线程,来更有效的利用服务器资源,满足需求更复杂,请求更多的场景。

(5)NIO是Netty的基础,读者可以多手动编写一下NIO的实现,来更深的了解Netty。


http://lihuaxi.xjx100.cn/news/284460.html

相关文章

假如曹操是一名程序员,会发生什么?

点击上方蓝色“视学算法”,选择“设为星标” 来源 | W3CSchool 东汉网络科技有限公司,本来是一家名扬四海的家族企业,可由于近几年来,越来越多的亲戚,在公司担任重要岗位,真正的人才越来越少,东…

这样就能用MathType编辑^符号

大家都知道数学公式中的符号有很多,有些符号的名称还很多,比如,^这个字符,可以是乘方、插入符号、插入符、托字符等。所以一些用户在使用过程中有点搞不清,但是Mathtype的符号模板有很多种,基本可以满足各种…

何恺明的GN之后,权重标准化新方法能超越GN、BN吗? | 技术头条

点击上方↑↑↑蓝字关注我们~「2019 Python开发者日」,购票请扫码咨询 ↑↑↑作者 | Siyuan Qiao、Huiyu Wang、Chenxi Liu、Wei Shen、Alan Yuille(Johns Hopkins University,约翰霍普金斯大学)译者 | 刘畅编辑 | Jane出品 | AI科…

如何在 Unix 和 DOS 格式之间转换文本文件

本文介绍如何在 Unix 和 DOS 格式之间转换文本文件。DOS 文本文件带有回车符( \r )和换行符( \n )作为它们的换行符,而 Unix 文本文件只有( \n )换行符作为换行符。 有多种方法可以将 DOS 文本文…

【leetcode】148. Sort List

Sort a linked list in O(n log n) time using constant space complexity. 链表排序可以用很多方法,插入,冒泡,选择都可以,也容易实现,但是复杂度不符合题意要求。 然后时间复杂度在O(nlogn)的排序算法中,…

南开大学提出目标检测新Backbone网络模块:Res2Net | 技术头条

点击上方↑↑↑蓝字关注我们~「2019 Python开发者日」,购票请扫码咨询 ↑↑↑作者 | 高尚华、程明明等(南开大学)译者 | 刘畅编辑 | Jane出品 | AI科技大本营(id:rgznai100)【导读】去年,AI科技…

GCC 参数列举及解释

GCC(GNU Compiler Collection,GNU编译器套件),是由 GNU 开发的编程语言编译器。它是以GPL许可证所发行的自由软件,也是 GNU计划的关键部分。gcc 与 g 分别是 gnu 的 c & c 编译器 gcc/g 在执行编译工作的时候,总共需要4步&…