Nettyの源码分析

本篇为Netty系列的最后一篇,按照惯例会简单介绍一些Netty相关核心源码。

1、Netty启动源码分析

        代码就使用最初的Netty服务器案例,在bind这一行打上断点,观察启动的全过程:

        由于某些方法的调用链过深,节约篇幅,不会一张张地截图,只会对最终结果或者关键部分进行说明分析:

        doBind 是一个重点方法,其中包含了:

  • initAndRegister:初始化ServerSocketChannel并将ServerSocketChannel注册到selector的方法
  • doBind0:ServerSocketChannel绑定端口号的方法

      

        1.1、initAndRegister

        首先会调用channelFactory工厂类的方法得到一个Channel

        相当于NIO中的:

ServerSocketChannel ssc = ServerSocketChannel.open();

        然后进入init方法,关键点在于,利用刚刚得到的channel对象,创建了一个流水线,并且添加了一个ChannelInitializer 处理器,监听初始化事件,在初始化事件中,使用eventLoop所在的NIO线程,提交一个任务,向pipeline中新增一个ServerBootstrapAcceptor用于处理新连接。

        真正的调用时机是在AbstractChannelregister0pipeline.invokeHandlerAddedIfNeeded();方法调用时被执行


        然后进入.register(channel)

        经过一系列的调用链,最终会进入AbstractChannelregister 方法:

        关键点:首先会判断当前线程是否是NIO线程,此时是主线程,所以会进入else分支:

        在try代码块中,会进行线程切换,由NIO线程负责注册。 

        我们选择NIO线程,进入register0方法,在register0方法中,有三个关键方法doRegister()pipeline.invokeHandlerAddedIfNeeded();safeSetSuccess(promise);

       1.1.1、 doRegister()

        上图中框出的这一行代码,相当于NIO中的

 SelectionKey sscKey = ssc.register(selector, 0, attach);

        pipeline.invokeHandlerAddedIfNeeded(); 该方法被执行时会回调ServerBootstrapinit方法的p.addLast,

        1.1.2、safeSetSuccess(promise);

        该方法是给主线程的final ChannelFuture regFuture 结果。参数中的promise和主线程的regFuture 是同一个。

        1.2、doBind0

        doBind0实际上是主线程注册的一个regFuture监听回调对象中的方法。当initAndRegister 返回结果后,才会触发回调对象中的operationComplete方法:

         在方法内部依旧是保证任务由NIO所在线程执行:

        经过一系列的调用,找到了AbstractChannel中的bind方法,bind方法中又有两个重点:

        doBind进行端口号绑定

        对应NIO中的:

 ssc.bind(new InetSocketAddress(8080));

        然后会进入if代码块判断,如果目前ServerSocketChannel处于Active状态,就触发流水线上所有的active事件。

        最后定位到AbstractChannel中的doBeginRead方法,在方法中注册一个接受连接事件:

        相当于NIO中的

 sscKey.interestOps(SelectionKey.OP_ACCEPT);

        小结:

        Netty的启动流程大致可以分为三部分:

  • 创建ServerSocketChannel对象。
  • 将ServerSocketChannel对象注册到selector上。
  • ServerSocketChannel进行端口绑定。

        其中,创建ServerSocketChannel对象是由主线程进行的,在将ServerSocketChannel对象注册到selector上时,会进行线程切换,由NIO线程去完成注册以及后续的端口绑定。

        在创建ServerSocketChannel对象后,会向ServerSocketChannel的流水线上先注册一个ChannelInitializer事件,加入acceptor handler,但是是在第二步注册后调用流水线的invokeHandlerAddedIfNeeded触发。

        端口绑定的方法dobind是regFuture的回调,第二步注册后会向promise中存放结果,由NIO所在线程执行端口绑定,绑定完成后触发NioServerSocketChannel的active事件,设置关注连接事件。


2、EventLoop源码分析

        在翻源码之前,我们简单地复习一下什么是EventLoop:

        EventLoop是一个不断循环的线程,用于处理所有注册到其上的事件。每一个Channel在创建时会被分配到一个EventLoop上,并且与其绑定,后续该Channel的所有事件都由这个EventLoop进行处理。

        EventLoop既可以处理IO事件,也可以处理普通事件或定时事件。

        我们重点看它的NioEventLoop实现,NioEventLoop 主要由selector,线程,任务队列组成。

        2.1、selector何时被创建

        NioEventLoop 有两个selector,可以理解成selector是经过封装优化的,而unwrappedSelector是原始的selector。

        它们是在构造方法中被初始化 :

       
        2.2、NioEventLoop 的NIO线程何时启动?

        通过下面的案例代码,观察NIO线程启动的时机:

public class TestEventLoop {
    public static void main(String[] args) {
        EventLoop eventLoop = new NioEventLoopGroup().next();
        eventLoop.execute(()->{
            System.out.println("test");
        });
    }
}

        进入execute方法:

        首先if代码块会检查任务对象是否为空。

        然后通过inEventLoop(); 方法检查当前线程是否是NIO线程,此时false。

        进入startThread()方法,第一次的state必然和ST_NOT_STARTED相等,进入最外层的if块。如果此时没有其他线程修改状态,则通过第二个if块中的CAS操作将状态修改为2,并且进入doStartThread()方法

        doStartThread()方法是启动NIO线程的核心方法:

        在 SingleThreadEventExecutor.this.run();中,会根据不同的事件执行对应的操作:

        如果没有任务,会进入SelectStrategy.SELECT分支,陷入阻塞。

        NIO线程是懒加载的,只有在执行execute方法时才会被创建。

        2.3、提交普通任务会不会结束select阻塞?

        在select方法内部,会调用有时限的阻塞方法,默认时间是1000ms,在这个期间如果被唤醒则会解除阻塞。

        在提交任务的execute中,有一个wakeup方法,我们选择它的NIO实现:

        如果不是当前NIO线程的任务,并且CAS成功(因为唤醒操作只需要调用一次wakeup方法,如果多个线程同时调用多次和调用一次的效果是一样的,多次调用影响性能。),才会调用唤醒方法:

        2.4、循环时什么时候会进入SelectStrategy.SELECT分支?

        进入SelectStrategy.SELECT分支的情况是没有任务:

        如果有任务会调用selectNowSupplier的get()方法返回一个selectNow()

        selectNow() 方法的作用是立刻查看selector上有无IO事件,如果有则会将IO事件也一起拿到,如果没有就返回0。

         

        2.5、NIO空轮询bug的体现以及Netty的解决方法

        什么是NIO的空轮询bug?指的是selector.select(timeoutMillis); 没有到超时时间,期间也没有任务或者事件,但是NIO线程没有在这一行陷入阻塞,而是不断地进行空循环。

        这种bug主要是出现在linux环境下,在Netty框架中对其进行了解决:

        关键点在于引入了一个计数器,每循环一次计数器+1

        当设置了阈值并且循环的次数超过了阈值,就可以认为发生了这个bug,会调用selectRebuildSelector 方法重建一个selector,并且将原有的key以及事件复制过去

 

        阈值的默认值是512次,或者通过参数进行设置:

        2.6、ioRatio的作用

        在NioEventLoop的run方法中有一段关于处理IO事件和普通事件的逻辑:

        其中涉及到了ioRatio,它在成员变量中的默认值是50。

        如果它的值为100,则会执行所有的IO事件和普通事件

        否则会对执行普通任务的时间进行计算,用当前时间 - IO事件发生前的当前时间 = IO事件的消耗时间。 假设为4s,然后用 4 * (100 - 50)/ 50 = 4s,得到普通任务的执行时间也为4s,如果在规定的时间内没有执行完普通任务,则会停止执行。

        2.7、selectedKeys优化

        在创建selector时,会通过反射将 Selector 实现类中的就绪事件集合替换为 SelectedSelectionKeySet

        SelectedSelectionKeySet 底层为数组实现,可以提高遍历性能:

        

        这一行是取出附件,在将ServerSocketChannel绑定到selector时,附件对象是Channel。

final Object a = k.attachment();

        满足下面的if代码块判断,会进入processSelectedKey(k, (AbstractNioChannel) a); 方法,在这个方法里会根据不同的事件类型做出判断并且执行:

3、accpet源码分析

        在原先的NIO中,一旦有事件发生,则会执行以下的代码逻辑:

//1 阻塞直到事件发生
selector.select();

Iterator<SelectionKey> iter = selector.selectedKeys().iterator();
while (iter.hasNext()) {    
    //2 拿到一个事件
    SelectionKey key = iter.next();
    
    //3 如果是 accept 事件
    if (key.isAcceptable()) {
        
        //4 执行 accept
        SocketChannel channel = serverSocketChannel.accept();
        channel.configureBlocking(false);
        
        //5 关注 read 事件
        channel.register(selector, SelectionKey.OP_READ);
    }
    // ...
}

        我们来看一下上面的代码在Netty中的实现过程:

        接着2.7中的代码,进入最后一个分支的read方法:

       选择AbstractNioMessageChannel 实现,关键代码有以下三处

       

        3.1、doReadMessages(readBuf)

        我们选择NioServerSocketChannel的实现:

 

        在SocketChannel ch = SocketUtils.accept(javaChannel()); 这一行代码中,会得到一个SocketChannel,相当于:

SocketChannel channel = serverSocketChannel.accept();

 

        然后会把这个SocketChannel封装在一个NioSocketChannel对象中,并且存放在参数中的list,然后返回:

        在NioSocketChannel父类的构造方法中也会设置channel为非阻塞,相当于:

channel.configureBlocking(false);

         

        3.2、allocHandle.incMessagesRead(localRead);

        这个方法的主要作用是接受客户端的连接。

        3.3、pipeline.fireChannelRead(readBuf.get(i));

        这个方法的作用是触发 read 事件,让 pipeline 上的 handler 处理,触发的是ServerBootstrapAcceptor 上的channelRead 事件:

         主要看try中的代码:

        又回到了register方法中,不过这次线程切换是从NIO的Boss切换到worker

        切换到worker线程:

        进入doRegister

        注册事件,相当于:

channel.register(selector, 0);

        最后关注read事件:

         一路跳转到doBeginRead中,执行关注read事件的逻辑:

        大致流程和accpet类似,最大的区别是由Worker线程完成。

4、read源码分析

       当客户端连接上服务器并且触发了一次write操作时,服务器首先会触发连接操作:

        跳过,下一次会触发读取操作:

         config.getAllocator(); 会分配一个ByteBufAllocator

         得到byteBuf:

        在循环中进行读取的逻辑:

 

 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Linux内核链表使用方法

简介&#xff1a; 链表是linux内核中最简单&#xff0c;同时也是应用最广泛的数据结构。内核中定义的是双向链表。 linux的链表不是将用户数据保存在链表节点中&#xff0c;而是将链表节点保存在用户数据中。linux的链表节点只有2个指针(pre和next)&#xff0c;这样的话&#x…

在Linux操作系统使用逻辑卷的快照(snapshot),进行对逻辑卷的数据备份。

作用&#xff1a;结合特定应用程序&#xff0c;方便备份数据。 基于cow&#xff08;copy on write 写时复制&#xff09;机制 在创建逻辑卷快照的时候&#xff0c;如果不去设置逻辑卷快照的权限的话&#xff0c;那么这个逻辑卷的权限就是可读可写&#xff0c; 创建逻辑卷快照…

coco数据集格式计算mAP的python脚本

目录 背景说明COCOeval 计算mAPtxt文件转换为coco json 格式自定义数据集标注 背景说明 在完成YOLOv5模型移植&#xff0c;运行在板端后&#xff0c;通常需要衡量板端运行的mAP。 一般需要两个步骤 步骤一&#xff1a;在板端批量运行得到目标检测结果&#xff0c;可保存为yol…

AI教你如何系统的学习Python

Python学习计划 第一阶段&#xff1a;Python基础&#xff08;1-2个月&#xff09; 目标&#xff1a;掌握Python的基本语法、数据类型、控制结构、函数、模块和包等。 学习Python基本语法&#xff1a;包括变量、数据类型&#xff08;整数、浮点数、字符串、列表、元组、字典、…

STM32基础篇:GPIO

GPIO简介 GPIO&#xff1a;即General Purpose Input/Output&#xff0c;通用目的输入/输出。就是一种片上外设&#xff08;内部模块&#xff09;。 对于STM32的芯片来说&#xff0c;周围有一圈引脚&#xff0c;有时需要对引脚进行读写&#xff08;读&#xff1a;从外部输入一…

【xinference】(15):在compshare上,使用docker-compose运行xinference和chatgpt-web项目,配置成功!!!

视频演示 【xinference】&#xff08;15&#xff09;&#xff1a;在compshare上&#xff0c;使用docker-compose运行xinference和chatgpt-web项目&#xff0c;配置成功&#xff01;&#xff01;&#xff01; 1&#xff0c;安装docker方法&#xff1a; #!/bin/shdistribution$(…

【嵌入式DIY实例-ESP8266篇】-LCD ST7735显示BMP280传感器数据

LCD ST7735显示BMP280传感器数据 文章目录 LCD ST7735显示BMP280传感器数据1、硬件准备与接线2、代码实现本文介绍如何将 ESP8266 NodeMCU 板 (ESP-12E) 与 Bosch Sensortec 的 BMP280 气压和温度传感器连接。 NodeMCU 微控制器 (ESP8266EX) 从 BMP280 传感器读取温度和压力值,…

VUE3初学入门-02-VUE创建项目

创建VUE项目的另一个方法 三种方法通过vue-cli进行创建通过npm进行创建比较 部署到nginx修改配置生成部署文件 三种方法 上一篇是在VSCODE中建立工作区&#xff0c;然后创建&#xff0c;属于命令加鼠标方式。个人感觉&#xff0c;在VSCODE基本上都是这样的操作&#xff0c;不是…

vue3中svg图标的封装与使用

组件封装&#xff1a; <template><svg :class"svgClass" :style"{ width: size px, height: size px, color: color, verticalAlign:deviationem}" aria-hidden"true"><use :xlink:href"#icon-${name}" /></s…

Python编程学习笔记(2)--- 列表简介

1、列表是什么 列表由一系列按特定顺序排列的元素组成。可以创建包含字母表中所有字母、数字、0~9或所有家庭成员姓名的列表&#xff1b;也可以将任何东西加入列表中&#xff0c;其中的元素之间可以没有任何关系。列表通常包含多个元素&#xff0c;因此给列表指定一个表示复数…

基于SSM+JSP的KTV点歌系统(带1w+文档)

基于SSMJSP的KTV点歌系统(带1w文档) 开发一个KTV点歌系统可以解决不利于线下点歌的问题&#xff0c;同时管理员可以利用网络对KTV点歌系统信息进行管理&#xff0c;设计的网站保证信息的完整安全&#xff0c;这样才能提高工作效率&#xff0c;保证系统安全正常的运行。 项目简介…

vim未找到命令,且yum install vim安装vim失败

vim未找到命令&#xff0c;且yum安装vim失败 1、wget更新yum云资源&#xff0c;本次更新为华为云镜像资源 wget -O /etc/yum.repos.d/CentOS-Base.repo https://mirrors.huaweicloud.com/repository/conf/CentOS-7-anon.repowget报未找到命令&#xff0c;请查看文章Linux wget…

iOS UITableView自带滑动手势和父视图添加滑动手势冲突响应机制探索

场景 我们有时候会遇到这样的一个交互场景&#xff1a;我们有一个UITableView 放在一个弹窗中&#xff0c;这个弹窗可以通过滑动进行展示和消失&#xff08;跟手滑动的方式&#xff09;&#xff0c;然后这个UITableView放在弹窗中&#xff0c;并且可以滚动&#xff0c;展示一些…

昇思25天学习打卡营第19天|Diffusion扩散模型

学AI还能赢奖品&#xff1f;每天30分钟&#xff0c;25天打通AI任督二脉 (qq.com) Diffusion扩散模型 本文基于Hugging Face&#xff1a;The Annotated Diffusion Model一文翻译迁移而来&#xff0c;同时参考了由浅入深了解Diffusion Model一文。 本教程在Jupyter Notebook上成…

Python数据分析案例50——基于EEMD-LSTM的石油价格预测

案例背景 很久没更新时间序列预测有关的东西了。 之前写了很多CNN-LSTM&#xff0c;GRU-attention&#xff0c;这种神经网络之内的不同模型的缝合&#xff0c;现在写一个模态分解算法和神经网络的缝合。 虽然eemd-lstm已经在学术界被做烂了&#xff0c;但是还是很多新手小白或…

RAG 案框架(Qanything、RAGFlow、FastGPT、智谱RAG)对比

各家的技术方案 有道的QAnything 亮点在&#xff1a;rerank RAGFLow 亮点在&#xff1a;数据处理index 智谱AI 亮点在文档解析、切片、query改写及recall模型的微调 FastGPT 优点&#xff1a;灵活性更高 下面分别按照模块比较各框架的却别 功能模块QAnythingRAGFLowFastG…

MPC学习资料汇总

模型预测控制MPC学习资料汇总 需要的私信我~ 需要的私信我~ 需要的私信我~ 【01】课件内容 包含本号所有MPC课程的课件&#xff0c;以及相关MATLAB文档。 【02】课件源代码 本号所有MPC课程的源代码。 【03】MPC仿真案例 三个MPC大型仿真案例&#xff1a; 1&#xff09;…

力扣爆刷第160天之TOP100五连刷66-70(回溯、旋转图像、技巧题)

力扣爆刷第160天之TOP100五连刷66-70&#xff08;回溯、旋转图像、技巧题&#xff09; 文章目录 力扣爆刷第160天之TOP100五连刷66-70&#xff08;回溯、旋转图像、技巧题&#xff09;一、110. 平衡二叉树二、39. 组合总和三、543. 二叉树的直径四、470. 用 Rand7() 实现 Rand1…

win系统提示VCRUNTIME140_1.dll丢失或找不到的8个处理方法

在使用电脑过程中经常会遇到各种各样的问题&#xff0c;比如vcruntime140_1.dll丢失或找不到vcruntime140_1.dll无法继续执行代码就是其中的一个常见问题!那么遇到vcruntime140_1.dll丢失问题要怎么处理&#xff1f;vcruntime140_1.dll是什么&#xff1f;下面我给大家详细介绍v…

谷粒商城学习笔记-16-人人开源搭建后台管理系统

文章目录 一&#xff0c;克隆前/后端代码1&#xff0c;克隆前端工程renren-fast-value2&#xff0c;克隆后端工程renren-fast 二&#xff0c;集成后台管理系统的后端代码三&#xff0c;启动后台管理系统四&#xff0c;前端系统的安装和运行1&#xff0c;下载安装VSCode2&#x…