Netty Review - ServerBootstrap源码解析

文章目录

  • 概述
  • 源码分析
  • 小结

在这里插入图片描述

在这里插入图片描述


概述

在这里插入图片描述

   ServerBootstrap bootstrap = new ServerBootstrap();
            bootstrap.group(bossGroup, workerGroup)
                    .channel(NioServerSocketChannel.class)
                    .option(ChannelOption.SO_BACKLOG, 1024)
                    .childHandler(new ChannelInitializer<SocketChannel>() {
                        @Override
                        protected void initChannel(SocketChannel ch) throws Exception {
                            ChannelPipeline pipeline = ch.pipeline();
                            //加入特殊分隔符分包解码器
                            pipeline.addLast(new DelimiterBasedFrameDecoder(10240, Unpooled.copiedBuffer("_".getBytes())));
                            //向pipeline加入解码器
                            pipeline.addLast("decoder", new StringDecoder());
                            //向pipeline加入编码器
                            pipeline.addLast("encoder", new StringEncoder());
                            //加入自己的业务处理handler
                            pipeline.addLast(new ChatServerHandler());
                        }
                    });

这段代码使用Netty构建了一个服务器。

  1. ServerBootstrap bootstrap = new ServerBootstrap(); - 创建一个ServerBootstrap实例,用于启动服务器。

  2. bootstrap.group(bossGroup, workerGroup) - 指定了服务器使用的两个EventLoopGroup,分别是bossGroup和workerGroup。其中,bossGroup用于接受传入的连接,而workerGroup用于处理已接受连接的流量。

  3. bootstrap.channel(NioServerSocketChannel.class) - 指定了服务器的Channel类型为NioServerSocketChannel,这表示使用NIO进行网络通信。

  4. bootstrap.option(ChannelOption.SO_BACKLOG, 1024) - 设置服务器的配置选项。在这里,设置了SO_BACKLOG,表示服务器套接字的连接队列大小为1024。

  5. bootstrap.childHandler(new ChannelInitializer<SocketChannel>() {...}) - 指定了当一个新的连接被接受时,所要执行的ChannelInitializer。这个匿名内部类用于配置新接受的Channel的ChannelPipeline,即为每个新接受的连接设置处理器。

  6. pipeline.addLast(new DelimiterBasedFrameDecoder(10240, Unpooled.copiedBuffer("_".getBytes()))) - 向ChannelPipeline中添加了一个DelimiterBasedFrameDecoder,用于根据特殊分隔符进行分包解码,这里的特殊分隔符是下划线"_”。

  7. pipeline.addLast("decoder", new StringDecoder()) - 向ChannelPipeline中添加了一个StringDecoder,用于将接收到的ByteBuf解码为字符串。

  8. pipeline.addLast("encoder", new StringEncoder()) - 向ChannelPipeline中添加了一个StringEncoder,用于将字符串编码为ByteBuf。

  9. pipeline.addLast(new ChatServerHandler()) - 向ChannelPipeline中添加了一个ChatServerHandler,这是自定义的业务处理Handler,用于处理接收到的消息。

总的来说,这段代码创建了一个基于Netty的服务器,配置了服务器的事件处理流程,包括接受连接、解码、编码和业务处理。


源码分析

ServerBootstrap bootstrap = new ServerBootstrap();

ServerBootstrap类的构造函数。在这个构造函数中,没有参数,它是一个默认构造函数。

bootstrap.group(bossGroup, workerGroup)

这段代码是ServerBootstrap类中的group方法的实现。

/**
 * Set the {@link EventLoopGroup} for the parent (acceptor) and the child (client). These
 * {@link EventLoopGroup}'s are used to handle all the events and IO for {@link ServerChannel} and
 * {@link Channel}'s.
 */
public ServerBootstrap group(EventLoopGroup parentGroup, EventLoopGroup childGroup) {
    // 调用父类的group方法,设置父EventLoopGroup
    super.group(parentGroup);

    // 检查子EventLoopGroup是否为空,如果为空则抛出NullPointerException
    if (childGroup == null) {
        throw new NullPointerException("childGroup");
    }

    // 检查是否已经设置了子EventLoopGroup,如果已经设置了则抛出IllegalStateException
    if (this.childGroup != null) {
        throw new IllegalStateException("childGroup set already");
    }

    // 将传入的子EventLoopGroup赋值给成员变量childGroup
    this.childGroup = childGroup;

    // 返回ServerBootstrap实例,用于链式调用
    return this;
}

这段代码的作用是设置ServerBootstrap的父(acceptor)和子(client)EventLoopGroup。这两个EventLoopGroup分别用于处理服务器端(acceptor)和客户端(client)的事件和IO操作。具体来说,这个方法会将传入的父EventLoopGroup设置给父类AbstractBootstrap,并将传入的子EventLoopGroup赋值给ServerBootstrap的成员变量childGroup。

在方法的实现中,

  • 首先调用了父类的group方法来设置父EventLoopGroup。
  • 然后,检查传入的子EventLoopGroup是否为空,如果为空则抛出NullPointerException。接着,检查是否已经设置了子EventLoopGroup,如果已经设置了则抛出IllegalStateException。
  • -最后,将传入的子EventLoopGroup赋值给成员变量childGroup,并返回ServerBootstrap实例,以支持链式调用。

总的来说,这段代码的目的是为ServerBootstrap设置父和子EventLoopGroup,以便于处理服务器和客户端的事件和IO操作,并提供了异常处理机制以确保参数的有效性。


我们看下 调用父类的group方法,设置父EventLoopGroup

super.group(parentGroup);

这段代码是一个泛型方法,通常用于在Netty的Bootstrap或ServerBootstrap中设置用于处理事件的EventLoopGroup。

/**
 * The {@link EventLoopGroup} which is used to handle all the events for the to-be-created
 * {@link Channel}
 */
public B group(EventLoopGroup group) {
    // 检查传入的EventLoopGroup是否为空,如果为空则抛出NullPointerException
    if (group == null) {
        throw new NullPointerException("group");
    }

    // 检查是否已经设置了EventLoopGroup,如果已经设置了则抛出IllegalStateException
    if (this.group != null) {
        throw new IllegalStateException("group set already");
    }

    // 将传入的EventLoopGroup赋值给成员变量group
    this.group = group;

    // 返回调用该方法的实例,以支持链式调用
    return self();
}

这个方法主要用于设置用于处理事件的EventLoopGroup,它会将传入的EventLoopGroup赋值给成员变量group。

在方法的实现中,

  • 首先检查传入的EventLoopGroup是否为空,如果为空则抛出NullPointerException。
  • 然后,检查是否已经设置了EventLoopGroup,如果已经设置了则抛出IllegalStateException。
  • 最后,将传入的EventLoopGroup赋值给成员变量group,并返回调用该方法的实例,以支持链式调用。

总的来说,这段代码的作用是为Netty的Bootstrap或ServerBootstrap设置EventLoopGroup,以便于处理事件,并提供了异常处理机制以确保参数的有效性。


channel(NioServerSocketChannel.class)

这段代码是一个泛型方法,用于设置用于创建Channel实例的Class对象。

/**
 * The {@link Class} which is used to create {@link Channel} instances from.
 * You either use this or {@link #channelFactory(io.netty.channel.ChannelFactory)} if your
 * {@link Channel} implementation has no no-args constructor.
 */
public B channel(Class<? extends C> channelClass) {
    // 检查传入的channelClass是否为空,如果为空则抛出NullPointerException
    if (channelClass == null) {
        throw new NullPointerException("channelClass");
    }

    // 调用channelFactory方法,传入一个ReflectiveChannelFactory实例,该实例用于通过反射创建Channel实例
    return channelFactory(new ReflectiveChannelFactory<C>(channelClass));
}

这个方法主要用于设置用于创建Channel实例的Class对象。它接受一个Class对象作为参数,并将其传递给channelFactory方法。

在这个方法内部,会创建一个ReflectiveChannelFactory实例,并将传入的Class对象作为参数传递给它。

在方法的实现中,

  • 首先检查传入的channelClass是否为空,如果为空则抛出NullPointerException。
  • 然后,创建一个ReflectiveChannelFactory实例,并将传入的Class对象作为参数传递给它。
  • 最后,调用channelFactory方法,将ReflectiveChannelFactory实例传递给它,并返回调用该方法的实例,以支持链式调用。

总的来说,这段代码的作用是为Netty的Bootstrap或ServerBootstrap设置用于创建Channel实例的Class对象,并提供了异常处理机制以确保参数的有效性。


new ReflectiveChannelFactory<C>(channelClass)

这段代码是ReflectiveChannelFactory类的构造函数实现。它接受一个Class对象作为参数,并使用反射机制获取该类的公共无参数构造方法。让我们逐步解释它:

public ReflectiveChannelFactory(Class<? extends T> clazz) {
    // 检查传入的clazz是否为空,如果为空则抛出NullPointerException
    ObjectUtil.checkNotNull(clazz, "clazz");

    try {
        // 使用反射获取传入的类的公共无参数构造方法
        this.constructor = clazz.getConstructor();
    } catch (NoSuchMethodException e) {
        // 如果获取构造方法失败,则抛出IllegalArgumentException异常
        throw new IllegalArgumentException("Class " + StringUtil.simpleClassName(clazz) +
                " does not have a public non-arg constructor", e);
    }
}

这个构造函数的作用是初始化ReflectiveChannelFactory实例。

它接受一个Class对象作为参数,该Class对象表示要实例化的Channel类。在构造函数内部,

  • 首先检查传入的Class对象是否为空,如果为空则抛出NullPointerException。
  • 然后,使用反射机制尝试获取传入类的公共无参数构造方法。如果获取构造方法失败,则抛出IllegalArgumentException异常,指示传入的类没有公共无参数构造方法。

总的来说,这段代码的作用是为ReflectiveChannelFactory类创建一个实例,并在构造函数中使用反射机制获取要实例化的Channel类的构造方法。

在这里插入图片描述

@Override
public T newChannel() {
    try {
        // 使用之前获取的构造方法实例化新的Channel对象
        return constructor.newInstance();
    } catch (Throwable t) {
        // 如果实例化过程中出现异常,则抛出ChannelException异常
        throw new ChannelException("Unable to create Channel from class " + constructor.getDeclaringClass(), t);
    }
}


/**
 * @deprecated Use {@link #channelFactory(io.netty.channel.ChannelFactory)} instead.
 */
@Deprecated
public B channelFactory(ChannelFactory<? extends C> channelFactory) {
    // 检查传入的channelFactory是否为空,如果为空则抛出NullPointerException
    if (channelFactory == null) {
        throw new NullPointerException("channelFactory");
    }
    // 检查是否已经设置了channelFactory,如果已经设置了则抛出IllegalStateException
    if (this.channelFactory != null) {
        throw new IllegalStateException("channelFactory set already");
    }

    // 将传入的channelFactory赋值给成员变量channelFactory
    this.channelFactory = channelFactory;
    return self();
}

在这里插入图片描述


option(ChannelOption.SO_BACKLOG, 1024)

这段代码定义了一个方法,用于为创建的Channel实例设置ChannelOption。

/**
 * Allow to specify a {@link ChannelOption} which is used for the {@link Channel} instances once they got
 * created. Use a value of {@code null} to remove a previous set {@link ChannelOption}.
 */
public <T> B option(ChannelOption<T> option, T value) {
    // 检查传入的option是否为空,如果为空则抛出NullPointerException
    if (option == null) {
        throw new NullPointerException("option");
    }
    // 如果value为空,则从options中移除之前设置的option
    if (value == null) {
        synchronized (options) {
            options.remove(option);
        }
    } else {
        // 否则,将option和对应的value放入options中
        synchronized (options) {
            options.put(option, value);
        }
    }
    // 返回调用该方法的实例,以支持链式调用
    return self();
}

这个方法的作用是允许指定一个ChannelOption,该选项在创建Channel实例后使用。如果value为null,则表示要移除之前设置的ChannelOption。否则,将option和对应的value放入options中。

在方法的实现中,

  • 首先检查传入的option是否为空,如果为空则抛出NullPointerException。
  • 然后,如果value为null,则从options中移除之前设置的option;否则,将option和对应的value放入options中。
  • 最后,返回调用该方法的实例,以支持链式调用。

这个方法的灵活性在于它允许用户根据需要设置或删除特定的ChannelOption,以满足不同场景的需求。

在这里插入图片描述


childHandler()

这段代码定义了一个方法,用于设置用于处理连接到ServerBootstrap的每个子Channel的ChannelHandler。

/**
 * Set the {@link ChannelHandler} which is used to serve the request for the {@link Channel}'s.
 */
public ServerBootstrap childHandler(ChannelHandler childHandler) {
    // 检查传入的childHandler是否为空,如果为空则抛出NullPointerException
    if (childHandler == null) {
        throw new NullPointerException("childHandler");
    }
    // 将传入的childHandler赋值给成员变量childHandler
    this.childHandler = childHandler;
    // 返回ServerBootstrap实例,以支持链式调用
    return this;
}

这个方法的作用是设置用于处理连接到ServerBootstrap的每个子Channel的ChannelHandler。传入的ChannelHandler将会被添加到每个新创建的子Channel的ChannelPipeline中,用于处理该子Channel的所有事件。

在方法的实现中,

  • 首先检查传入的childHandler是否为空,如果为空则抛出NullPointerException。
  • 然后,将传入的childHandler赋值给成员变量childHandler。
  • 最后,返回ServerBootstrap实例,以支持链式调用。

这种设计模式允许用户通过链式调用一系列方法来配置ServerBootstrap的参数,从而更加简洁和灵活地构建Netty服务器。


小结

ServerBootstrap是Netty中用于创建服务器端应用程序的引导类。它的设计目的是提供一种简洁、灵活的方式来配置和启动服务器,并处理与客户端的连接。

以下是ServerBootstrap的设计要点总结:

  1. 引导配置链式调用:ServerBootstrap类提供了一系列方法,允许用户通过链式调用来配置服务器的各种参数,如设置EventLoopGroup、Channel类型、Channel选项、ChannelHandler等。这种设计模式使得配置过程更加简洁和灵活。

  2. EventLoopGroup的配置:通过group方法,用户可以设置用于处理服务器端连接和客户端连接的EventLoopGroup。通常,一个用于接受连接的bossGroup和一个用于处理连接请求的workerGroup会被设置。

  3. Channel类型的设置:用户可以通过channel方法设置用于创建Channel实例的类型,例如NioServerSocketChannel.class。这决定了服务器将使用的底层传输协议。

  4. Channel选项的设置:option/childOption方法允许用户为创建的Channel实例设置各种选项,如SO_BACKLOG、TCP_NODELAY等。

  5. ChannelHandler的配置:通过childHandler方法,用户可以设置用于处理连接到服务器的每个子Channel的ChannelHandler。这些ChannelHandler将被添加到每个新创建的子Channel的ChannelPipeline中,用于处理子Channel的所有事件。

  6. 灵活性和可扩展性:ServerBootstrap的设计允许用户根据具体需求灵活地配置服务器的各种参数,同时也提供了可扩展的接口和回调机制,使得用户可以根据需要自定义处理逻辑。

总的来说,ServerBootstrap的设计通过提供一系列简洁而灵活的配置方法,以及可扩展的接口和回调机制,使得用户能够轻松地构建高性能、可定制的服务器应用程序。
在这里插入图片描述

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

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

相关文章

Stable Diffusion 模型下载:DreamShaper(梦想塑造者)

本文收录于《AI绘画从入门到精通》专栏&#xff0c;专栏总目录&#xff1a;点这里。 文章目录 模型介绍生成案例案例一案例二案例三案例四案例五案例六案例七案例八案例九案例十 下载地址 模型介绍 DreamShaper 是一个分格多样的大模型&#xff0c;可以生成写实、原画、2.5D 等…

STM32 USART串口通信

目录 USART串口 串口发送 串口发送接收 串口收发HEX数据包 串口收发文本数据包 USART串口 串口发送 Serial.c #include "stm32f10x.h" // Device header #include "stdio.h" #include "stdarg.h"/*** brief 初始化串口以…

C++ 广度优先搜索(bfs)(五十四)【第一篇】

今天我们来学习一下一个新的搜索&#xff0c;广度优先搜索。 1.广度优先搜索的前提 队列&#xff08;queue&#xff09; 是一种 操作受限制 的线性表&#xff0c;其限制&#xff1a; 只允许从表的前端&#xff08;front&#xff09;进行删除操作&#xff1b; 只允许在表的后端…

openssl3.2 - osslsigncode工程的学习

文章目录 openssl3.2 - osslsigncode工程的学习概述笔记工程库地址工程的编译osslsigncodeM工程文件列表osslsigncodeM工程搭建细节原始工程实现的改动自己封装的包含openssl和curl的实现osslsigncodeM工程命令行的用法备注 - VS2019调试环境备注 - 如果要单步openssl的API学学…

OpenCV-37 最小外接矩形和最大外接矩形

一、外接矩形 外接矩形分为最小外接矩形和最大外接矩形。 下图中红色矩形为最小外接矩形&#xff0c;绿色矩形为最大外接矩形。 1. 最小外接矩形 minAreaRect(points) --- 最小外接矩形 point为轮廓&#xff1b; 返回值为元组&#xff0c;内容是一个旋转矩形(RotatedRect…

MySQL简单配置GTID

前期规划 IP地址 角色 系统版本 内核 软件包名称 192.168.2.3 Mysql主服务器 CentOS Stream 9 5.14.0- 381.el9.x86_64 mysql-8.2.0-linux-glibc2.17-x86_64.tar.xz 192.168.2.4 Mysql从服务器 CentOS Stream 9 5.14.0- 381.el9.x86_64 mysql-8.2.0-linux-glibc…

【Tauri】(2):使用Tauri应用开发,使用开源的Chatgpt-web应用做前端,使用rust 的candle做后端,本地运行小模型桌面应用

视频演示地址 https://www.bilibili.com/video/BV17j421X7Zc/ 【Tauri】&#xff08;2&#xff09;&#xff1a;使用Tauri应用开发&#xff0c;使用开源的Chatgpt-web应用做前端&#xff0c;使用rust 的candle做后端&#xff0c;本地运行小模型桌面应用 1&#xff0c;做一个免…

MIT-Missing Semester_Topic 3:Editors (Vim) 练习题

文章目录 练习一练习二练习三练习四练习五练习六练习七练习八 本 Topic 的 MIT 讲解网页&#xff08;练习题未给解答&#xff09; 练习一 自行完成 vimtutor。vimtutor 是 Vim 本身附带的一个入门教程&#xff0c;在 shell 中直接输入 vimtutor 便能运行。注意该教程在 8024 大…

代码随想录day20--二叉树的应用8

LeetCode669.修剪二叉搜索树 题目描述&#xff1a; 给你二叉搜索树的根节点 root &#xff0c;同时给定最小边界low 和最大边界 high。通过修剪二叉搜索树&#xff0c;使得所有节点的值在[low, high]中。修剪树 不应该 改变保留在树中的元素的相对结构 (即&#xff0c;如果没…

vector容器

1. vector基本概念 1.1 功能&#xff1a; vector数据结构和数组非常相似&#xff0c;也称为单端数组 vector与普通数组区别&#xff1a; 不同之处在于数组是静态空间&#xff0c;而vector可以动态扩展 动态扩展&#xff1a; 并不是在原空间之后续接新空间&#xff0c;而是找更…

【蓝桥杯Python】试题 算法训练 比较

资源限制 内存限制&#xff1a;256.0MB C/C时间限制&#xff1a;1.0s Java时间限制&#xff1a;3.0s Python时间限制&#xff1a;5.0s 问题描述 给出一个n长的数列&#xff0c;再进行m次询问&#xff0c;每次询问询问两个区间[L1,R1]&#xff0c;[L2,R2]&#xff0c;   …

C语言特殊指针

1 char型指针 char型指针实质上跟别的类型的指针并无本质区别&#xff0c;但由于C语言中的字符串以字符数组的方式存储&#xff0c;而数组在大多数场合又会表现为指针&#xff0c;因此字符串在绝大多数场合就表现为char型指针。 定义&#xff1a; char *p "abcd"…

爬爬爬——今天是浏览器窗口切换和给所选人打钩(自动化)

学习爬虫路还很长&#xff0c;第一阶段花了好多天了&#xff0c;还在底层&#xff0c;虽然不是我专业要学习的语言&#xff0c;和必备的知识&#xff0c;但是我感觉还挺有意思的。加油&#xff0c;这两天把建模和ai也不学了&#xff0c;唉过年了懒了&#xff01; 加油坚持就是…

ChatGPT高效提问—prompt实践(绘画、Logo设计)

ChatGPT高效提问—prompt实践&#xff08;绘画、logo设计&#xff09; 1.1 绘画工具 ​ Midjourney是一款由Midjourney研究实验室研发的人工智能程序&#xff0c;可根据文本生成图像。你可以通过浏览器在聊天应用程序Discord中向Midjourney机器人发送消息来使用它。不需要安装…

前端 > JS 笔试题面试考题(21-25)

简述请看下面的代码片段并回答以下问题 &#xff1f; for (var i 0; i< 5; i){var btn document.createElement(button);btn.appendChild(document.createTextNode(Button i));btn.addEventListener(click, function(){ console.log(${i} );});document.body.appendChild…

网络专栏目录

大家好我是苏麟 , 这是网络专栏目录 . 图解网络 资料来源 : 小林coding 小林官方网站 : 小林coding (xiaolincoding.com) 图解网络目录 基础篇 基础篇 TCP/IP网络模型有几层? : TCP/IP网络模型 键入网址到页面显示,期间发生了什么? : 键入网址到页面显示,期间发生了什么 现阶…

2024-02-12 Unity 编辑器开发之编辑器拓展3 —— EditorGUI

文章目录 1 GUILayout2 EditorGUI 介绍3 文本、层级、标签、颜色拾取3.1 LabelField3.2 LayerField3.3 TagField3.4 ColorField3.5 代码示例 4 枚举选择、整数选择、按下按钮4.1 EnumPopup / EnumFlagsField4.2 IntPopup4.3 DropdownButton4.4 代码示例 5 对象关联、各类型输入…

【Docker】Docker Container(容器)

文章目录 一、什么是容器&#xff1f;二、为什么需要容器&#xff1f;三、容器的生命周期容器OOM容器异常退出容器暂停 四、容器命令详解docker createdocker logsdocker attachdocker execdocker startdocker stopdocker restartdocker killdocker topdocker statsdocker cont…

「数据结构」哈希表1:基本概念

&#x1f387;个人主页&#xff1a;Ice_Sugar_7 &#x1f387;所属专栏&#xff1a;Java数据结构 &#x1f387;欢迎点赞收藏加关注哦&#xff01; 基本概念 &#x1f349;哈希表&#x1f349;哈希冲突&#x1f34c;负载因子调节&#x1f34c;解决哈希冲突&#x1f95d;1. 闭散…

HCIA-HarmonyOS设备开发认证V2.0-3.2.轻量系统内核基础-中断管理

目录 一、中断基础概念二、中断管理使用说明三、中断管理模块接口四、代码分析&#xff08;待续...&#xff09; 一、中断基础概念 在程序运行过程中&#xff0c;出现需要由 CPU 立即处理的事务时&#xff0c;CPU 暂时中止当前程序的执行转而处理这个事务&#xff0c;这个过程…