网络编程进化史:Netty Channel 的崭新篇章

news2025/1/31 12:39:23

上篇文章(Netty 入门 — ByteBuf,Netty 数据传输的载体),我们了解了 Netty 的数据是以 ByteBuf 为单位进行传输的,但是有了数据,你没有通道,数据是无法传输的,所以今天我们来熟悉 Netty 的第三个核心组件:Channel。ByteBuf 是数据,那 Channel 则是负责传输数据的通道,它是把握 Netty 通信的命门,没有它 Netty 是无法通信的。

Channel 简介

在 Java NIO 中我们知道,Channel,即通道,是用来传输数据的一条“管道”,它与 Buffer 相辅相成,在 Java NIO 中,我们只能从 Channel 读取数据到 Buffer 中,或者从 Buffer 读取数据到 Channel 中,如下:

在 Netty 中同样有一个 Channel,该 Channel 是 Netty 的核心概念之一,它是 Netty 网络 IO 操作的抽象,即 Netty 网络通信的主体,由它来负责对端进行网络通信、注册、数据操作等一切 IO 相关的操作,其主要功能包括:

  1. 网络 IO 的读写
  2. 客户端发起连接
  3. 关闭连接
  4. 网络连接的相关参数
  5. 绑定端口
  6. Netty 框架相关操作,如获取 Channel 相关联的 EventLoop、pipeline 等。

为什么要另起炉灶?

JDK 提供了一个 Channel,为什么 Netty 还要另起炉灶自己实现一个呢?我认为主要原因有如下几个:

  1. 原生的 Channel 功能太少,不满足 Netty 的要求。
  2. 原生的 ServerSocketChannel 和 SocketChannel 是一个 SPI 接口,具体的实现由虚拟厂商来实现,直接通过原生 ServerSocketChannel 和 SocketChannel 来实现及满足 Netty 的要求,其工作量不亚于重新开发一个。
  3. Netty 的 Channel 需要符合 Netty 的整体架构设计,他需要和 Netty 的整体架构耦合在一起,比如 IO 模型、基于元数据描述配置化的TCP参数等等,原生的 Channel 都不支持。
  4. 自定义的 Channel,灵活性更高,功能更加全面。

Channel 原理

Channel 的核心原理如下图:

  1. 客户端与服务端成功建立连接后,服务端会为该连接创建一个 Channel。
  2. Channel 从 EventLoopGroup 中获取一个 EventLoop,Channel 注册到该 EventLoop 中,从此 Channel 就与该 EventLoop 绑定在一起了,在 Channel 整个生命周期内都只会与该 EventLoop 绑定在一起。
  3. 客户端发起的 IO 操作,在 Channel 中都将产生相对应的 Event,触发与该 Channel 绑定的 EventLoop 进行处理
  4. 如果是读写事件,执行线程调度 ChannelPipeline 来处理业务逻辑。ChannelPipeline 只负责 Handler 的编排,真正执行任务的是各个具体的 ChannelHandler。

Channel 的状态转换

Channel 从创建到消亡,他有四种状态,分别是:

  1. 打开状态(Open)
    1. Channel 处于打开状态时,表示它已经被创建,但尚未绑定到任何地址或连接到远端服务器。
  2. 活动状态(Active)
    1. Channel 处于活动状态时,表示它已经成功绑定到本地地址或连接到远端服务器。
    2. 这个时候可以调用 writeAndFlush() 向对方发送数据了。
  3. 非活动状态(Inactive)
    1. Channel 处于非活动状态时,表示它已经处于活动状态,但连接已经断开或由于其他原因不可用。
    2. 当连接被关闭或出现错误时,Channel 会进入非活动状态。
    3. 无法进行读取或写入操作,但可重新激活 Channel。
  4. 关闭状态(Closed):
    1. Channel 处于关闭状态时,表示它已经完全关闭,无法再进行任何操作。

状态流转如下:

Netty 提供了四个方法来判断 Channel 的状态:

  • isOpen():检查 Channel 是否为 open 状态。
  • isRegistered():检查 Channel 是否为 registered 状态。
  • isActive():检查 Channel 是否为 active 状态。
  • isWritable():这个方法有误导性,它并不是判断当前 Channel 是否可写,实际上它是用来检测当前 Channel 的写操作是否可以立刻被 IO 线程处理,当该方法返回 false 时,任何写请求都会被阻塞,知道 I/O 线程有能力能处理这些请求。

各个状态以及他们对应的操作如下表格:

状态isOpen()isActive()close()writeAndFlush()读操作写操作
打开(Open)truefalsetruefalsetruetrue
活动(Active)truetruetruetruetruetrue
非活动(Inactive)truefalsetruefalsefalsefalse
关闭(Closed)falsefalsefalsefalsefalsefalse

Channel 的 API

Channel 常用的 API 非常多,如下图(部分):

方法虽然多,但是总体大致分为如下几类:

类 getter API

这里方法主要用于获取 Channel 相关的属性,如绑定地址,相关配置等等

  • SocketAddress localAddress():返回与 Channel 绑定的本地地址
  • SocketAddress remoteAddress():返回与 Channel 绑定的远端地址
  • ChannelConfig config():返回一个 ChannelConfig 对象,通过这个对象可以配置Channel相关的参数
  • ChannelMetadata metadata():返回一个 ChannelMetadata 对象,ChannelMetadata 可以查询 Channel 实现是否支持某种操作,目前它还只要一个方法 hasDisconnect(),用来判断 Channel 实现是否支持 disconnect() 操作。
  • Channel parent():返回 Channel 的 parent Channel。SocketChannel 返回的是相对一个的 ServerSocketChannel,而 ServerSocketChannel 则返回 null。为什么 SocketChannel 返回的是 ServerSocketChannel 呢?因为所有的 SocketChannel (客户端发起连接)都是由 ServerSocketChannel 接受连接而创建的,所以 SocketChannel 的 parent() 返回的就是对应的 ServerSocketChannel 。
  • EventLoop eventLoop():返回 Channel 注册的 EventLoop。
  • ChannelPipeline pipeline():返回与 Channel 关联的 ChannelPipeline。
  • ByteBufAllocator alloc():返回与 Channel 关联的 ByteBufAllocator 对象。
  • Unsafe unsafe():返回 Channel 的 Unsafe 对象。Unsafe 是 Channel 的内部类,只供 Channel 内部使用。
Future 相关 API

Netty 中的所有 IO 操作都是异步的,这就意味着任何的 IO 调用都将立刻返回,但是并不能保证所有的操作都在调用结束后就完成了,而且我们也不知道 IO 操作执行的结果。Netty 在完成 IO 调用后会返回一个 Future 对象,该对象就是 Channel 异步 IO 的结果。Channel 提供了我们操作这些 Future 的方法:

  • ChannelFuture closeFuture():当 Channel 关闭时返回一个 ChannelFuture,我们可以通过该方法来来对 Channel 关闭后做一些处理。
  • ChannelPromise voidPromise():返回一个 ChannelPromise 实例对象。
  • ChannelPromise newPromise():返回一个 ChannelPromise。
  • ChannelProgressivePromise newProgressivePromise():返回一个新的 ChannelProgressivePromise 实例对象。
  • ChannelFuture newSucceededFuture():创建一个新的 ChannelFuture,并将其标注为 succeed 。
  • ChannelFuture newFailedFuture(Throwable cause):创建一个新的 ChannelFuture,并将其标注为 failed。

ChannelFutureChannelPromise 是 Netty 提供的两个特殊 Future,利用他们我们能够在 Netty 完成一些异步操作的处理。

判断状态 API

Channel 提供了四个 isXxx 方法用来判断 Channel 的状态:

  • boolean isOpen():判断 Channel 是否 opened
  • boolean isRegistered():判断 Channel 是否 registered
  • boolean isActive():判断 Channel 是否 active
  • boolean isWritable():判断 Channel 是否可以立刻处理 IO 事件
事件触发类方法

这些方法都会触发 IO 事件,他们都会通过 ChannelPipeline 传播然后被 ChannelHandler 处理。

  • ChannelFuture bind(SocketAddress localAddress):服务端绑定本地端口,开始监听客户端的连接请求。
  • ChannelFuture connect(SocketAddress remoteAddress):客户端向服务端发起连接请求。
  • ChannelFuture disconnect():断开连接,但是需要注意的是该方法不会释放资源,它还可以再次通过 connect() 再次与服务端建立连接。
  • ChannelFuture close():关闭通道,释放资源。
  • Channel read():读取通道
  • ChannelFuture write(Object msg):向 Channel 中写入数据,该方法并不会将数据真实地写入通道,它只将数据写入到了通断缓存区,我们需要调用 flush() 将缓存区的数据刷入到 Channel 中。
  • Channel flush():将数据刷写到 Channel 中。
  • ChannelFuture writeAndFlush(Object msg):相当于调用了 write()flush()

Channel 的配置

ChannelConfig

在 Netty 中,每个 Channel 都有与之相对应的 ChannelConfig , 可以通过调用 config() 来获取。ChannelConfig 是一个接口,每个特定的 Channel 都有具体的 ChannelConfig 实现类,例如:

  • NioSocketChannel 的对应的配置类为 NioSocketChannelConfig。
  • NioServerSocketChannel 的对应的配置类为 NioServerSocketChannelConfig。

整体的 UML 图如下:

具体的实现我们这篇文章不需要关系,我们需要关注的是它提供了哪些 Config。

  • ChannelConfig 提供通用型配置

    • ChannelOption.CONNECT_TIMEOUT_MILLIS:连接超时时间,默认值30000毫秒即30秒。
    • ChannelOption.WRITE_SPIN_COUNT:写操作的最大循环数,即一次写事件处理期间最多调用 write() 的次数。它有点儿像 Java 中的自旋锁。引入该参数的主要木的是为了避免一个 Channel 写入大量数据,对其他网络通道的读写处理带来延时。
    • ChannelOption.ALLOCATOR:设置内存分配器。
    • ChannelOption.RCVBUF_ALLOCATOR:对读事件设置内存分配器。
    • ChannelOption.AUTO_READ:配置是否自动触发 read() ,默认为 True,程序不需要显示调用 read()
    • ChannelOption.AUTO_CLOSE:配置当写事件失败时,是否自动关闭 Channel,默认为 True。
    • ChannelOption.WRITE_BUFFER_HIGH_WATER_MARK:设置写缓存区的高水位线。如果写缓存区中的数据超过该值, Channel#isWritable() 方法将返回 false。
    • ChannelOption.WRITE_BUFFER_LOW_WATER_MARK:设置写缓存区的低水位线。如果写缓存区的数据超过高水位线后,通道将变得不可写,等写缓存数据降低到低水位线后通道恢复可写状态(Channel#isWritable()将再次返回true)。
    • ChannelOption.MESSAGE_SIZE_ESTIMATOR:设置用于检测通道消息大小的检测器:MessageSizeEstimator。

    这里引入了两个概念:高水位线和低水位线,这两个概念我们在讲缓冲区的时候再细说。

  • NioSocketChannelConfig

    NioSocketChannelConfig 在 ChannelConfig 的基础上增加了如下几个配置:

    • ChannelOption.SO_KEEPALIVE: 连接保持,默认为 False,我们可以将这个参数视为 TCP 的心跳机制。
    • ChannelOption.SO_REUSEADDR:地址复用,默认值False。
    • ChannelOption.SO_LINGER:关闭 Socket 的延迟时间,默认值为 -1,表示禁用该功能
    • ChannelOption.TCP_NODELAY:立即发送数据,默认值为 Ture。该值其实是设置 Nagle 算法的启用。关于 Nagle 算法我们后面再细说。
    • ChannelOption.SO_RCVBUF:TCP 数据接收缓冲区大小。该缓冲区即 TCP 接收滑动窗口。
    • ChannelOption.SO_SNDBUF:TCP 数据发送缓冲区大小。该缓冲区即 TCP 发送滑动窗口。
    • ChannelOption.IP_TOS:IP 参数,设置 IP 头部的 Type-of-Service 字段,用于描述 IP 包的优先级和 QoS 选项。
    • ChannelOption.ALLOW_HALF_CLOSURE:一个连接的远端关闭时本地端是否关闭,默认值为False。
  • NioServerSocketChannelConfig

    • ChannelOption.SO_REUSEADDR:地址复用,默认值False。
    • ChannelOption.SO_RCVBUF:TCP 数据接收缓冲区大小。该缓冲区即 TCP 接收滑动窗口。
    • ChannelOption.SO_BACKLOG:服务端接受连接的队列长度,如果队列已满,客户端连接将被拒绝。

从上面我们可以看到 ChannelConfig 提供的都是一些通用型的配置,而 NioSocketChannelConfig 和 NioServerSocketChannelConfig 提供的基本上都是 Socket 相关的配置参数,每个都与 java.net.StandardSocketOptions 定义的标准 TCP 参数一一对应。

由于这个是入门篇,所以这里大明哥就不再扩展阐述了,对这些配置参数更加详细的说明,大明哥后面会专门有文章类分析的,这里我们只需要了解 ChannelConfig 是我们配置 Channel 通道相关参数的服务类即可。

Channel 的使用方法

看完上面部分,大明哥相信你对 Channel 有了一个基本的了解。其实Channel 的 API 没啥好演示的,因为这些 API 都不是单独使用的,需要一些其他的组件来配合,但是咱们还是要有仪式感对吧,就写一个简单的 demo 来看看 Channel 的状态变化以及简单感受下异步的风采。

  • 服务端
public static void main(String[] args) throws InterruptedException {
  Channel channel = new ServerBootstrap()
                  .group(new NioEventLoopGroup())
                  .channel(NioServerSocketChannel.class)
                  .childHandler(new ChannelInitializer<NioSocketChannel>() {
                      @Override
                      protected void initChannel(NioSocketChannel ch) throws Exception {
                          ch.pipeline().addLast(new LoggingHandler());
                      }
                  })
                  .bind(8081)
                  .channel();
  System.out.println("isOpen:" + channel.isOpen() + ";;;isRegistered:" + channel.isRegistered() + ";;;isActive:" + channel.isActive());

  System.out.println("eventLoop():" + channel.eventLoop());
  System.out.println("pipeline():" + channel.pipeline());

  TimeUnit.SECONDS.sleep(5);
  System.out.println("=============================");
  System.out.println("isOpen:" + channel.isOpen() + ";;;isRegistered:" + channel.isRegistered() + ";;;isActive:" + channel.isActive());
}
  • 客户端
public static void main(String[] args) throws InterruptedException {
    Channel channel = new Bootstrap()
            .group(new NioEventLoopGroup())
            .channel(NioSocketChannel.class)
            .handler(new ChannelInitializer<SocketChannel>() {
                @Override
                protected void initChannel(SocketChannel ch) throws Exception {
                    ch.pipeline().addLast(new LoggingHandler());
                }
            })
            .connect("127.0.0.1",8081)
            .channel();

    System.out.println("isOpen:" + channel.isOpen() + ";;;isRegistered:" + channel.isRegistered() + ";;;isActive:" + channel.isActive());

    System.out.println("eventLoop():" + channel.eventLoop());
    System.out.println("pipeline():" + channel.pipeline());

    TimeUnit.SECONDS.sleep(5);
    System.out.println("=============================");
    System.out.println("isOpen:" + channel.isOpen() + ";;;isRegistered:" + channel.isRegistered() + ";;;isActive:" + channel.isActive());
}
  • 运行结果
// server
isOpen:true;;;isRegistered:false;;;isActive:false
eventLoop():io.netty.channel.nio.NioEventLoop@2f333739
pipeline():DefaultChannelPipeline{(ServerBootstrap$1#0 = io.netty.bootstrap.ServerBootstrap$1)}
=============================
isOpen:true;;;isRegistered:true;;;isActive:true

//client
isOpen:true;;;isRegistered:false;;;isActive:false
eventLoop():io.netty.channel.nio.NioEventLoop@6ed3ef1
pipeline():DefaultChannelPipeline{(ChannelTestClient$1#0 = com.sike.netty.rumen.ChannelTestClient$1)}
=============================
isOpen:true;;;isRegistered:true;;;isActive:true

从结果中可以看出,无论是服务端还是客户端,Channel 都是异步的,当服务端调用 bind() 方法后返回的 Channel,它仅仅只完成了新建,注册以及绑定工作都没有完成,等待 5 秒后,我们再看其状态,则都是 true 了。

代码地址:http://m6z.cn/5O6hON

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

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

相关文章

2023 年专业人士必须尝试的 15 款人工智能工具

在本文中&#xff0c;我们将看到一些人工智能驱动的工具&#xff0c;这些工具将在 2023 年彻底改变开发并使专业人士的生活变得轻松。我们将讨论旨在为专业人士提供支持的 15 种顶级人工智能和低代码工具。人工智能工具现在变得更加强大&#xff0c;使他们能够创造有影响力的产…

数字图像处理(十六)非局部均值去噪

文章目录 一、前言二、NL-means1.两个邻域块的相似度2.NL-means原理3.数学理论推导4.代码链接 参考链接 一、前言 在之前我们已经介绍过许多图像去噪的方法&#xff0c;比如均值滤波、中值滤波、高斯滤波等。今天我们要介绍一种新的去噪方法——非局部均值去噪&#xff08;the…

如何构建一个外卖微信小程序

随着外卖行业的不断发展&#xff0c;越来越多的商家开始关注外卖微信小程序的开发。微信小程序具有使用方便、快速上线、用户覆盖广等优势&#xff0c;成为了商家们的首选。 那么&#xff0c;如何快速开发一个外卖微信小程序呢&#xff1f;下面就让我们来看看吧&#xff01; 首…

二、W5100S/W5500+RP2040树莓派Pico<DHCP>

文章目录 1 前言2 简介2 .1 什么是DHCP&#xff1f;2.2 为什么要使用DHCP&#xff1f;2.3 DHCP工作原理2.4 DHCP应用场景 3 WIZnet以太网芯片4 DHCP网络设置示例概述以及使用4.1 流程图4.2 准备工作核心4.3 连接方式4.4 主要代码概述4.5 结果演示 5 注意事项6 相关链接 1 前言 …

模型保存和加载

1、sklearn模型的保存和加载API from sklearn.externals import joblib 保存&#xff1a;joblib.dump(rf, ‘test.pkl’)加载&#xff1a;estimator joblib.load(‘test.pkl’) 2、决策树的模型保存加载案例 保存&#xff1a; import joblib from sklearn.model_selectio…

程序员想要网上接单却不知道如何是好?那这篇文章你可得收藏好了!

作为一名程序员&#xff0c;想要网上接单赚赚零花钱&#xff0c;提高生活水平&#xff0c;这当然是无可厚非的&#xff0c;甚至有许多人已经将网上接单作为主业。 可是面对网上五花八门的接单平台&#xff0c;看着网上真真假假的信息&#xff0c;你真的清楚如何选择吗&#xf…

CAP定理下:Zookeeper、Eureka、Nacos简单分析

CAP定理下&#xff1a;Zookeeper、Eureka、Nacos简单分析 CAP定理 C: 一致性&#xff08;Consistency&#xff09;&#xff1a;写操作之后的读操作也需要读到之前的 A: 可用性&#xff08;Availability&#xff09;&#xff1a;收到用户请求&#xff0c;服务器就必须给出响应 P…

整个自动驾驶小车001:概述

材料&#xff1a; 1&#xff0c;树梅派4b&#xff0c;作为主控&#xff0c;这个东西有linux系统&#xff0c;方便 2&#xff0c;HC-S104超声波模块&#xff0c;我有多个&#xff0c;不少于4个&#xff0c;我可以前后左右四个方向都搞一个 3&#xff0c;l298n模块&#xff0c;…

指定顺序输出

系列文章目录 进阶的卡莎C++_睡觉觉觉得的博客-CSDN博客数1的个数_睡觉觉觉得的博客-CSDN博客双精度浮点数的输入输出_睡觉觉觉得的博客-CSDN博客足球联赛积分_睡觉觉觉得的博客-CSDN博客大减价(一级)_睡觉觉觉得的博客-CSDN博客小写字母的判断_睡觉觉觉得的博客-CSDN博客纸币(…

SpringCloud复习:(5) feign的底层原理

首先需要在启动类加注解EnableFeignClients,它的代码如下&#xff1a; 可以看到&#xff0c;EnableFeignClients导入了FeignClientRegistra类&#xff0c;它实现了ImportBeanDefinitionRegistrar&#xff0c;其中定义了注册bean的逻辑 其中registerFeignClients用于注册我们通…

YOLO目标检测——行人数据集【含对应voc、coco和yolo三种格式标签+划分脚本】

实际项目应用&#xff1a;智能监控、人机交互、行为分析、安全防护数据集说明&#xff1a;行人检测数据集&#xff0c;真实场景的高质量图片数据&#xff0c;数据场景丰富标签说明&#xff1a;使用lableimg标注软件标注&#xff0c;标注框质量高&#xff0c;含voc(xml)、coco(j…

gcc和makfile

gcc和makfile gcc预处理(进行宏替换)编译(生成汇编)汇编(生成机器可以识别的代码)连接(生成可以执行的文件或者库文件) makemakefile文件的编写 gcc 没安装的话&#xff0c;按照终端提示的安装命令安装就行 运行的格式&#xff1a; gcc [选项] [要编译的文件] [选项] [目标文件…

✔ ★【备战实习(面经+项目+算法)】 10.25学习(算法刷题:5道)

✔ ★【备战实习&#xff08;面经项目算法&#xff09;】 坚持完成每天必做如何找到好工作1. 科学的学习方法&#xff08;专注&#xff01;效率&#xff01;记忆&#xff01;心流&#xff01;&#xff09;2. 每天认真完成必做项&#xff0c;踏实学习技术 认真完成每天必做&…

html2pdf

页面布局时将需要保存在同一页pdf的dom元素用div包裹&#xff0c;并为该div添加class类名&#xff0c;例如.convertPDF&#xff0c;如果有多页创建多个.convertPDF这个div&#xff0c;再循环保存pdf即可 用到了html2canvas和JsPdf这两个插件&#xff0c;自行站内搜索安装

使用 jdbc 技术升级水果库存系统

抽取执行更新方法抽取查询方法 —— ResultSetMetaData ResultSetMetaData rsmd rs.getMetaData();//元数据,结果集的结构数据 抽取查询方法 —— 解析结果集封装成实体对象提取 获取连接 和 释放资源 的方法将数据库配置信息转移到配置文件 <dependencies><depend…

【Win11】系统重装教程(最新最详细)

目录 一.简介 二.用U盘制作PE系统 三、安装系统 软件&#xff1a;Windows 11版本&#xff1a;21H2语言&#xff1a;简体中文大小&#xff1a;5.14G安装环境&#xff1a;PE系统&#xff0c;至少7代处理器硬件要求&#xff1a;CPU2.0GHz 内存4G(或更高&#xff09;下载通道①丨…

10 款超强大性能测试工具,你用过哪几款呢?

这篇关于“性能测试工具”的文章将按以下顺序让您了解不同的软件测试工具&#xff1a; 什么是性能测试&#xff1f; 为什么我们需要性能测试&#xff1f; 性能测试的优势 性能测试的类型 十大性能测试工具 什么是性能测试&#xff1f; 性能测试是一种软件测试&#xff0c;可确…

【Netty专题】用Netty手写一个远程长连接通信框架

目录 前言阅读对象阅读导航前置知识课程内容一、使用Netty实现一个通信框架需要考虑什么问题二、通信框架功能设计2.1 功能描述2.2 通信模型2.3 消息体定义2.4 心跳机制2.5 重连机制*2.6 Handler的组织顺序2.7 交互式调试 三、代码实现&#xff1a;非必要。感兴趣的自行查看3.1…

matlab中narginchk函数用法及其举例

matlab中narginchk函数用法及其举例 narginchk在编写子函数程序时候&#xff0c;在验证输入参数数目方面具有重要作用&#xff0c;本博文讲一讲该函数的用法。 一、narginchk功能 narginchk的作用是验证输入参数数目。 二、语法 narginchk(minArgs,maxArgs)narginchk(minA…