玩转Netty,从“Hello World”开始

news2024/12/25 23:53:50

大家好,我是老三,之前里,我们讨论了Java的三种IO模型,提到了网络通信框架Netty,它简化和优化了NIO的使用,这期,我们正式开始走近Netty。

为什么要用Netty?

首先当然是NIO的使用,本身比较复杂,而且还存在一些问题。

除此之外,如果在项目的开发中,要实现稳定的网络通信,就得考虑网络的闪断、客户端的重复接入、客户端的安全认证、消息的编解码、半包读写……

头大

所以,巧了,恰好有这么一个成熟稳定、性能强大、开箱即用的网络框架摆在我们面前,相比较Java NIO,Netty更加出色:

  • 易用性: Netty 在 NIO 基础上进行了更高层次的封装,屏蔽了 NIO 的复杂性,大大降低了开发难度;Netty 提供了很多开箱即用的工具,例如常用的行解码器、长度域解码器等,不需要自己再实现。
  • 稳定性: Netty 更加可靠稳定,修复和完善了 JDK NIO 较多已知问题,例如臭名昭著的 select 空转导致 CPU 消耗 100%,TCP 断线重连,keep-alive 检测等问题。
  • 可扩展性: Netty 的的可扩展性做的非常好,比如支持可定制化的线程模型。

我们有什么理由拒绝这么一款优秀的网络通信框架呢?代码怎么写不是写喽!

初识Netty

什么是Netty?

Netty官方是这么定义Netty的:

Netty 是一个异步事件驱动的网络应用程序框架,用于快速开发可维护的高性能协议服务器和客户端。

组成图-来源官方

  • Netty是一个开源的、单线程模型的 Java 网络编程框架。

  • Netty基于 NIO ,被广泛应用于各种网络应用程序开发。

  • Netty支持多种协议,包括但不限于 HTTP、WebSocket、TCP、UDP 和 SSL/TLS 协议等。

  • Netty 是非阻塞的,事件驱动的框架。

  • Netty具有高性能、可扩展和易于使用的优点。

Netty的现状?

Netty 由 JBoss 社区开发维护的,它的社区相对比较活跃:

  • https://github.com/netty/netty:Github已经收获31.2K星标
  • https://netty.io/:官方网站,提供了比较完整的文档

官方目前最新的版本是5.x,,但是很不幸,已经被社区放弃开发维护,属于废弃版本,最新的稳定版本是4.x 。

一般使用,推荐4.x,Netty 4.x对3.x不做兼容,我们后续的学习也基于Netty 4.x版本。

谁在用Netty?

作为最流行的网络通信框架,大量的公司选择它作为底层网络通信框架,包括不限于:

使用Netty的公司

我们可能自己没有直接用过Netty,但其实熟悉的很多开源中间件,都用到了Netty,比如:

  • 服务治理:Apache Dubbo、gRPC。
  • 大数据:Hbase、Spark、Flink、Storm。
  • 搜索引擎:Elasticsearch。
  • 消息队列:RocketMQ、ActiveMQ。

用到Netty的优秀产品非常多,大家感兴趣可以看看:https://netty.io/wiki/related-projects.html。

从"Hello World"开始

气氛衬托到这,不写个Demo也过不去,还是从"Hello World"开始,我们领略一下Netty的风采。

  1. 创建一个Maven项目:这个就不用多说了吧

创建Maven项目

  1. 导入依赖:我们直接用4.x最新的版本
        <dependency>
            <groupId>io.netty</groupId>
            <artifactId>netty-all</artifactId>
            <version>4.1.92.Final</version>
        </dependency>
  1. 编写代码:那么我们就开始编写这个Demo的服务器和客户端相关代码

    • NettyServer:基于Netty的客户端

      /**
       * <p>Date: 2023/5/14 10:29</p>
       * <p>Author: fighter3</p>
       * <p>Description: Netty服务端Demo</p>
       */
      public class NettyServer{
          // 服务器监听的端口号
          private int port;
      
          public NettyServer(int port) {
              this.port = port;
          }
      
          /**
           * 启动Netty服务器
           * @throws InterruptedException
           */
          public void run() throws InterruptedException {
              // 创建boss线程组和worker线程组
              // bossGroup 用于监听客户端的连接请求,将连接请求发送给 workerGroup 进行处理
              NioEventLoopGroup bossGroup = new NioEventLoopGroup();
              // workerGroup 用于处理客户端连接的数据读写
              NioEventLoopGroup workerGroup = new NioEventLoopGroup();
              try {
                  // 创建 ServerBootstrap 对象,用于启动 Netty 服务器
                  ServerBootstrap serverBootstrap = new ServerBootstrap();
                  // 绑定线程池事件组
                  serverBootstrap.group(bossGroup, workerGroup)
                          .channel(NioServerSocketChannel.class)
                          // 通道初始化回调函数,在启动的时候可以自动调用
                          .childHandler(new ChannelInitializer<SocketChannel>() {
                              @Override
                              public void initChannel(SocketChannel ch) throws Exception {
                                  ChannelPipeline pipeline = ch.pipeline();
                                  // 添加消息处理器
                                  pipeline.addLast(new NettyServerHandler());
                              }
                          });
      
                  // 绑定端口,开始接收客户端请求
                  ChannelFuture channelFuture = serverBootstrap.bind(port).sync();
      
                  System.out.println("Netty服务器监听端口:"+port);
      
                  // 等待服务端监听端口关闭
                  channelFuture.channel().closeFuture().sync();
              } finally {
                  //释放线程组资源
                  bossGroup.shutdownGracefully();
                  workerGroup.shutdownGracefully();
              }
          }
      
          public static void main(String[] args) throws InterruptedException {
              // 创建服务器对象,监听端口号为 8888
              NettyServer server = new NettyServer(8888);
              System.out.println("============Netty服务器启动...=============");
              // 启动服务器
              server.run();
              System.out.println("============Netty服务器停止...=============");
          }
      }
      
    • NettyServerHandler:服务器的消息处理器,用于处理各种事件

      /**
       * <p>Date: 2023/5/14 10:30</p>
       * <p>Author: fighter3</p>
       * <p>Description: Netty服务器消息处理器</p>
       */
      public class NettyServerHandler extends ChannelInboundHandlerAdapter {
      
          /**
           * 当客户端上线的时候会触发这个方法
           * @param ctx
           * @throws Exception
           */
          @Override
          public void channelActive(ChannelHandlerContext ctx) throws Exception {
              String message="你好,靓仔!";
              ByteBuf hello = Unpooled.copiedBuffer(message, CharsetUtil.UTF_8);
              // 发送消息
              ctx.writeAndFlush(hello);
          }
      
          /**
           *当 Channel 中有来自客户端的数据时就会触发这个方法
           */
          @Override
          public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
              ByteBuf buf = (ByteBuf) msg;
              System.out.println("客户端发来的消息:" + buf.toString(CharsetUtil.UTF_8)); // 接收消息并打印输出
          }
      
          /**
           * 当有异常时触发这个方法
           */
          @Override
          public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
              cause.printStackTrace();
              ctx.close();
          }
      }
      
  • NettyClient:使用Netty的客户端,通过ip和端口连接服务端

    /**
     * <p>Date: 2023/5/14 10:32</p>
     * <p>Author: fighter3</p>
     * <p>Description: Netty客户端Demo</p>
     */
    public class NettyClient {
        // 服务器 IP
        private String host;
        // 服务器监听的端口号
        private int port;
    
        public NettyClient(String host, int port) {
            this.host = host;
            this.port = port;
        }
    
        /**
         * 启动 Netty 客户端
         */
        public void run() throws InterruptedException {
            // 创建事件循环组
            NioEventLoopGroup group = new NioEventLoopGroup();
            try {
                // 创建 Bootstrap 对象
                Bootstrap bootstrap = new Bootstrap();
                // 配置 Bootstrap 对象
                // 设置线程组
                bootstrap.group(group)
                        // 设置客户端通信的通道类型为NIO类型
                        .channel(NioSocketChannel.class)
                        .handler(new ChannelInitializer<SocketChannel>() {
                            // 通道初始化回调函数,在启动的时候可以自动调用
                            @Override
                            public void initChannel(SocketChannel ch) throws Exception {
                                // 添加消息处理器
                                ch.pipeline().addLast(new NettyClientHandler());
                            }
                        });
                // 连接服务器,异步等待连接成功
                ChannelFuture channelFuture = bootstrap.connect(host, port).sync();
                System.out.println("===========Netty客户端连接服务端=========");
    
                // 等待客户端连接关闭
                channelFuture.channel().closeFuture().sync();
            } finally {
                //释放资源
                group.shutdownGracefully();
            }
        }
    
        public static void main(String[] args) throws InterruptedException {
           // 创建客户端对象,并连接到服务器
            NettyClient client = new NettyClient("127.0.0.1", 8888);
            // 启动客户端,开始发送消息
            client.run();
        }
    }
    
  • NettyClientHandler:Netty客户端处理器,用于处各种事件

    /**
     * <p>Date: 2023/5/14 10:33</p>
     * <p>Author: fighter3</p>
     * <p>Description: Netty客户端处理器</p>
     */
    public class NettyClientHandler extends ChannelInboundHandlerAdapter {
    
        /**
         * 当 Channel 准备就绪时就会触发这个方法
         */
        @Override
        public void channelActive(ChannelHandlerContext ctx) throws Exception {
            String message="大佬,带带我!";
            ByteBuf hello = Unpooled.copiedBuffer(message, CharsetUtil.UTF_8);
            // 发送消息
            ctx.writeAndFlush(hello);
        }
    
        /**
         * 当 Channel 中有来自服务器的数据时就会触发这个方法
         */
        @Override
        public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
            ByteBuf buf = (ByteBuf) msg;
            System.out.println("服务端发来的消息:" + buf.toString(CharsetUtil.UTF_8)); // 接收消息并打印输出
        }
    
        /**
         * 发生异常就会触发这个方法
         */
        @Override
        public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
            cause.printStackTrace();
            ctx.close();
        }
    }
    
  1. 运行一下:先启动NettyServer,再启动NettyClient,看下运行结果

    ============Netty服务器启动...=============
    Netty服务器监听端口:8888
    客户端发来的消息:大佬,带带我!
    
    ===========Netty客户端连接服务端=========
    服务端发来的消息:你好,靓仔!
    

好了,一个简单的Netty入门Demo就写完了,Netty是一个双工通信的网络框架,可以看到,服务端和客户端,流程基本上一致,主要包括这么几个步骤:

  1. 创建事件循环组和相关对象,用于监听和处理网络事件;
  2. 配置 Netty 服务器或客户端的启动参数,包括线程组、通道类型、TCP 参数等;
  3. 给服务器或客户端的 ChannelPipeline 添加各种 ChannelHandler,用于处理不同的网络事件;
  4. 绑定端口启动服务器或连接服务器;
  5. 等待服务器或客户端连接关闭,释放相关资源。

服务器&客户端初始化启动流程

虽然这个Demo比较简单,但其实已经用到了Netty里几个比较关键的组件:

  1. ByteBuf:Netty 的字节容器,类似于 Java 的 ByteBuffer,但是提供了更加强大、简便且安全的 API,用于在网络中传递二进制数据;
  2. EventLoopGroup:Netty 的事件循环组,用于管理和调度连接到服务器或者从服务器连接出去的所有 Channel 上的事件循环;
  3. ServerBootstrap:Netty 的服务器启动类,用于启动和配置一个 TCP/IP 服务器;
  4. Bootstrap:Netty 的客户端启动类,用于启动和配置一个 TCP/IP 客户端;
  5. Channel:Netty 的核心概念,用于表示一个通信通道,可以读取和写入数据;
  6. ChannelPipeline:Netty 的 Channel 处理器,用于在传入的数据上执行一组 ChannelHandler;
  7. ChannelHandler:Netty 的核心组件,用于处理各种通信事件,例如读取数据、写数据、建立连接等;

Netty的重要组件

后续,我们还会和这些组件打更多的交道。


好了,那么这期内容就到这了,这期里我们初步了解了Netty,包括什么是Netty、Netty现状、Netty的应用,还写了一个简单的Demo。下一期,我们继续深入了解Netty,敬请期待。



参考:

[1].https://netty.io/

[2].《Netty权威指南》

[3]. 《Netty核心原理剖析与RPC实践》

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

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

相关文章

知行之桥EDI系统2023版功能介绍——日志页面

在知行之桥EDI系统2023版中&#xff0c;除了在此前的文章中曾经介绍过的概览页面之外&#xff0c;还新增了日志页面。日志页面基于旧版本的状态页面进行了功能优化&#xff0c;为用户展示了消息、交易日志、应用程序日志、访问日志以及审计日志五种类型&#xff0c;每种日志类型…

MySQL 复合查询 内外连接

目录 基本查询回顾 多表查询 自连接 group by可以带多个 子查询 在from子句中使用子查询 合并查询 表的内连和外连 内连接 外连接 左外连接 右外连接 关于自连接和内连接&#xff08;chatgpt&#xff09; 基本查询回顾 查询工资高于 500 或岗位为 MANAGER 的雇员…

虚拟键盘:十六进制值和鼠标或键盘等效项。 代码按数字顺序列出

Virtual-Key Codes (Winuser.h) - Win32 apps | Microsoft Learn 虚拟键码 (Winuser.h) - Win32 apps | Microsoft Learn Value说明0x01鼠标左键0x02鼠标右键0x03控制中断处理0x04中间鼠标按钮 (三键鼠标)0x05X1 鼠标按钮0x06X2 鼠标按钮0x07Undefined0x08BACKSPACE 密钥0x09T…

注册阿里云OSS步骤

注册阿里云OSS步骤 阿里云是阿里巴巴集团旗下全球领先的云计算公司&#xff0c;也是国内最大的云服务提供商 。 云服务指的就是通过互联网对外提供的各种各样的服务&#xff0c;比如像&#xff1a;语音服务、短信服务、邮件服务、视频直播服务、文字识别服务、对象存储服务等等…

【SAM】CAN SAM COUNT ANYTHING? AN EMPIRICAL STUDY ON SAM COUNTING

论文链接&#xff1a; 代码链接&#xff1a; 目的 探索SAM在few-shot setting的object counting的能力。 结论 它目前落后于最先进的few-shot object counting方法&#xff0c;特别是对于小而拥挤的物体。两个主要原因。首先&#xff0c;SAM倾向于使用单个掩码分割同一类别…

javaIO流之序列流

目录 简介一、ObjectOutputStream二、ObjectInputStream三、Kryo四、小结 简介 Java 的序列流&#xff08;ObjectInputStream 和 ObjectOutputStream&#xff09;是一种可以将 Java 对象序列化和反序列化的流。 序列化是指将一个对象转换为一个字节序列&#xff08;包含对象的…

《算法训练营》贪心入门 + 10题

&#x1f442; 梦寻古镇 - 羽翼深蓝Wings - 单曲 - 网易云音乐 &#x1f442; 如果我有一个男朋友 - 于娜懿 - 单曲 - 网易云音乐 &#x1f442; 对酒&#xff08;女生版&#xff09; - 浅影阿 - 单曲 - 网易云音乐 &#x1f442; 知我&#xff08;抒情版&#xff09; - 尘a…

分享一个程序员接私活、兼职的平台

分享一个程序员接私活、兼职的平台 1、技术方向满足任一即可2、技术要求3、最后 1、技术方向满足任一即可 Python&#xff1a;熟练掌握Python编程语言&#xff0c;能够使用Python进行数据处理、机器学习和深度学习等相关工作。 MATLAB&#xff1a;熟练掌握MATLAB编程语言&…

API接口设计方案

API&#xff08;Application Programming Interface&#xff09;接口是一种用于与应用程序进行交互的标准化接口&#xff0c;它允许第三方应用程序通过网络调用应用程序的功能。设计API接口是开发人员在开发软件系统时的重要任务之一&#xff0c;因为API接口的质量和易用性直接…

UML的14种图

目录 1.UML 2.0包括14种图&#xff0c;分别列举如下&#xff1a; 2.活动图、用例图 3.部署图、顺序图 4.类图、协作图 5. 状态图、构件图 6.UML结构图和行为图 7.最后推荐一个图片转文字网站 8.最后推荐一个抖音去水印在线工具 9.最后推荐一个PPT模板下载网站 1.UML 2…

基于移动端vue的项目中webpack升级注意事项

这里写自定义目录标题 欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题&#xff0c;有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants 创建一个自定义列表如何创建一个…

Android应用-开发框架设计

目录 1. &#x1f4c2; 简介 1.1 背景 1.2 专业术语 2. &#x1f531; 总体设计思想 2.1 分层&#xff1a;组件化设计框架 2.2 分类&#xff1a;应用开发架构图 3. ⚛️ 框架详细设计 3.1 组件化框架外形 3.2 业务模块化 3.3 代码编程框架 4. &#x1f4a0; 框架其他…

streamlit

正常在学习一个新框架之前&#xff0c; 肯定要先调研下这个框架究竟能做些什么事吧&#xff1f; 但对于 streamlit 来说&#xff0c;请你相信我&#xff0c;这是一个你可以无脑去学习的框架&#xff0c;我之所以这么说&#xff0c;是因为我相信终有一天&#xff0c;你一定能用…

【算法学习系列】04 - 由01不等概率随机实现01等概率随机

文章目录 约定条件说明解决方案思路说明实现代码 验证方案 约定条件说明 存在函数 unequalP_01() 不等概率返回 0 和 1。只能通过函数 unequalP_01() 来实现 01 等概率随机函数 equalP_01() 解决方案 为了能够验证最后的等概率随机&#xff0c;这里先把函数 unequalP_01() 实现…

美颜SDK设计思路与架构分析:如何实现高可定制化

当下&#xff0c;美颜SDK也成为了移动应用开发中的必备工具之一。在实际应用中&#xff0c;不同的应用场景需要不同的美颜效果&#xff0c;因此如何实现高可定制化的美颜SDK就成为了一个重要的技术问题。 一、美颜SDK的设计思路 美颜SDK的设计思路需要考虑以下几个方面&…

Vivado 下 AD9767 双通道正弦波产生例程

Vivado 下 AD9767 双通道正弦波产生例程 1、实验简介 本实验基于 Xinlinx 黑金 AX7A035 FPGA 开发板&#xff0c; 练习使用 AN9767 模块&#xff0c;实验中使用的模块是采用 ANALOG DEVICES 公司的 AD9767 芯 片&#xff0c;支持独立双通道、14 位、 125MSPS 的数模转…

金币商城功能迭代. 使用版本号解决数据的并发修改问题

前言. 公司的商城模块嵌在微信公众号里面, 商城里面除了少量的现金业务, 大头在金币业务里面, 商城本来就是用来增加客户粘度的, 金币是客户通过某些行为免费获得如注册, 绑定,推荐等 需求. 金币方面之前的设计: 1.金币只有一个流水表,消费为负,获取为正 2.并且…

github上有什么好的node.js的项目?

前言 可以参考一下下面的nodejs相关的项目&#xff0c;希望对你的学习有所帮助&#xff0c;废话少说&#xff0c;让我们直接进入正题 1、 NodeBB Star: 13.3k 一个基于Node.js的现代化社区论坛软件&#xff0c;具有快速、可扩展、易于使用和灵活的特点。它支持多种数据库&am…

驱动开发:通过应用堆实现多次通信

在前面的文章《驱动开发&#xff1a;运用MDL映射实现多次通信》LyShark教大家使用MDL的方式灵活的实现了内核态多次输出结构体的效果&#xff0c;但是此种方法并不推荐大家使用原因很简单首先内核空间比较宝贵&#xff0c;其次内核里面不能分配太大且每次传出的结构体最大不能超…

jvm之GC

写在前面 本文一起看下GC相关的内容。 1&#xff1a;GC基础内容 1.1&#xff1a;为什么要有GC 内存资源的稀缺性&#xff0c;以及内存管理的复杂性&#xff0c;决定了需要有垃圾回收器这样的角色来帮助我们管理内存资源&#xff0c;避免手动管理带来的内存不能得到正常释放…