Java开发中Netty线程模型原理解析

Java开发中Netty线程模型原理解析,Netty是Java领域有名的开源网络库具有高性能和高扩展性的特点,很多流行的框架都是基于它来构建。Netty 线程模型不是一成不变的,取决于用户的启动参数配置。通过设置不同的启动参数Netty ,可同时支持 Reactor 单线程模型、多线程模型。

Netty线程模型就是Reactor模式的实现,如图:

Java开发中Netty线程模型原理解析

Netty抽象了两组线程池BossGroup和WorkerGroup,其类型都是NioEventLoopGroup,BossGroup用来接受客户端发来的连接WorkerGroup则负责对完成TCP三次握手的连接进行处理。

NioEventLoopGroup里面包含了多个NioEventLoop管理NioEventLoop的生命周期。每个NioEventLoop中包含了一个NIO Selector、一个队列、一个线程;其中线程用来做轮询注册到Selector上的Channel的读写事件和对投递到队列里面的事件进行处理。

Boss NioEventLoop线程的执行步骤:

(1)处理accept事件与client建立连接, 生成NioSocketChannel。

(2)将NioSocketChannel注册到某个worker NIOEventLoop上的selector

(3)处理任务队列的任务 即runAllTasks。

Worker NioEventLoop线程的执行步骤:

(1)轮询注册到自己Selector上的所有NioSocketChannel的read和write事件。

(2)2处理read和write事件在对应NioSocketChannel处理业务。

(3)#runAllTasks处理任务队列TaskQueue的任务,一些耗时的业务处理可以放入TaskQueue中慢慢处理这样不影响数据在pipeline中的流动处理。

Worker NIOEventLoop处理NioSocketChannel业务时,使用了pipeline (管道),管道中维护了handler处理器链表用来处理channel中的数据。

ChannelPipeline

Netty将Channel的数据管道抽象为ChannelPipeline,消息在ChannelPipline中流动和传递。ChannelPipeline持有I/O事件拦截器ChannelHandler的双向链表,由ChannelHandler对I/O事件进行拦截和处理,可以方便的新增和删除ChannelHandler来实现不同的业务逻辑定制不需要对已有的ChannelHandler进行修改能够实现对修改封闭和对扩展的支持。

ChannelPipeline是一系列的ChannelHandler实例,流经一个Channel的入站和出站事件可以被ChannelPipeline 拦截。每当一个新的Channel被创建了,都会建立一个新的ChannelPipeline并绑定到该Channel上,这个关联是永久性的;Channel既不能附上另一个ChannelPipeline也不能分离当前这个。这些都由Netty负责完成,而无需开发人员的特别处理。

根据起源一个事件将由ChannelInboundHandler或ChannelOutboundHandler处理,ChannelHandlerContext实现转发或传播到下一个ChannelHandler。一个ChannelHandler处理程序可以通知ChannelPipeline中的下一个ChannelHandler执行。Read事件(入站事件)和write事件(出站事件)使用相同的pipeline,入站事件会从链表head 往后传递到最后一个入站的handler出站事件会从链表tail往前传递到最前一个出站的 handler,两种类型的 handler 互不干扰。

Java开发中Netty线程模型原理解析

ChannelInboundHandler回调方法:

Java开发中Netty线程模型原理解析

ChannelInboundHandler回调方法

Java开发中Netty线程模型原理解析

异步非阻塞

写操作:通过NioSocketChannel的write方法向连接里面写入数据时候是非阻塞的,马上会返回即使调用写入的线程是我们的业务线程。Netty通过在ChannelPipeline中判断调用NioSocketChannel的write的调用线程是不是其对应的NioEventLoop中的线程,如果发现不是则会把写入请求封装为WriteTask投递到其对应的NioEventLoop中的队列里面,然后等其对应的NioEventLoop中的线程轮询读写事件时候,将其从队列里面取出来执行。

读操作:当从NioSocketChannel中读取数据时候并不是需要业务线程阻塞等待,而是等NioEventLoop中的IO轮询线程发现Selector上有数据就绪时,通过事件通知方式来通知业务数据已就绪,可以来读取并处理了。

每个NioSocketChannel对应的读写事件都是在其对应的NioEventLoop管理的单线程内执行,对同一个NioSocketChannel不存在并发读写,所以无需加锁处理。

使用Netty框架进行网络通信时,当我们发起I/O请求后会马上返回,而不会阻塞我们的业务调用线程;如果想要获取请求的响应结果,也不需要业务调用线程使用阻塞的方式来等待,而是当响应结果出来的时候,使用I/O线程异步通知业务的方式,所以在整个请求 -> 响应过程中业务线程不会由于阻塞等待而不能干其他事情。

本文来自投稿,不代表重蔚自留地立场,如若转载,请注明出处https://www.cwhello.com/263263.html

如有侵犯您的合法权益请发邮件951076433@qq.com联系删除

(0)
黑马程序员黑马程序员订阅用户
上一篇 2023年5月15日 09:08
下一篇 2023年5月15日 09:08

相关推荐

  • 我来教你c#中多线程。

    C#中多线程是实现并行处理的一种方式,可以提高程序的执行效率。 在C中,多线程是一种处理多个数据的有效方法,通过使用多线程,我们可以同时执行多个任务,从而提高程序的执行效率,本文将详细介绍如何在C中使用多…

    2024年7月12日
    04
  • 我来说说storm崩溃问题怎么解决。

    Storm是一个开源的分布式实时计算系统,被广泛应用于大数据处理、实时分析等领域,在使用过程中,可能会遇到Storm崩溃的问题,本文将介绍一些常见的Storm崩溃问题及其解决方法。 1. 内存不足导致崩溃 Storm在运行过…

    2024年6月13日
    07
  • 小编分享java内存模型包含什么内容。

    Java内存模型(JMM)是一种抽象的概念,它描述的一组规则或者规范,定义了程序中各个变量的访问方式。JMM主要解决了多线程环境中的可见性问题,即当一个共享变量被一个线程修改后,其他线程何时能看见该变化。除了…

    2024年7月12日
    04
  • 什么是关系模型? 关系模型的基本概念

    关系模型由IBM公司研究员Edgar Frank Codd于1970年发表的论文中提出,经过多年的发展,已经成为目前最常用、最重要的模型之一。在关系模型中有一些基本的概念,具体如下。(1)关系(Relation)。关系一词与数学领域有…

    2023年6月8日
    055
  • 说说七彩虹b365m主板配什么显卡。

    七彩虹B365M主板是一款基于Intel B365芯片组的微型ATX主板,适用于搭建入门级至中等性能的桌面电脑,在选择CPU时,需要确保所选处理器与主板兼容,同时符合用户的性能需求和预算,以下是针对七彩虹B365M主板搭配CPU…

    2024年6月21日
    029
  • 小编教你java中线程的状态有哪几种。

    Java 中的线程状态是 Java 多线程编程中的一个重要概念,线程在它的生命周期内会经历多种不同的状态,这些状态反映了线程的运行情况和可执行性,Java 语言定义了以下六种线程状态: 1、新建(New) 线程对象被创建…

    2024年7月18日
    07
  • 软件测试模型有几种?这4中软件测试模型你都知道吗

    在软件开发过程中,人们根据经验教训并结合未来软件的发展趋势总结出了很多软件开发模型,如瀑布模型、快速原型模型、迭代模型等,这些模型对软件开发过程具有很好的指导作用,但遗憾的是它们对软件测试并没有给予…

    2023年5月6日 编程分享
    05
  • yolo算法模型训练的样本怎样构建?

    在进行模型训练时,我们需要构造训练样本和设计损失函数,才能利用梯度下降对网络进行训练。训练样本的构建将一幅图片输入到yolo模型中,对应的输出是一个7x7x30张量,构建标签label时对于原图像中的每一个网格grid…

    2023年8月29日 编程技术分享
    05

联系我们

QQ:951076433

在线咨询:点击这里给我发消息邮件:951076433@qq.com工作时间:周一至周五,9:30-18:30,节假日休息