您当前的位置:首页 > 电脑百科 > 站长技术 > 服务器

玩转Netty,从“Hello World”开始!

时间:2023-05-16 15:43:18  来源:今日头条  作者:架构师之道

为什么要.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的服务器和客户端相关代码
  2. 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服务器停止...=============");
    }}
  3. 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,敬请期待。



Tags:Netty   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,不构成投资建议。投资者据此操作,风险自担。如有任何标注错误或版权侵犯请与我们联系,我们将及时更正、删除。
▌相关推荐
Netty入门实践:模拟IM聊天
我们使用的框架几乎都有网络通信的模块,比如常见的Dubbo、RocketMQ、ElasticSearch等。它们的网络通信模块使用Netty实现,之所以选择Netty,有两个主要原因: Netty封装了复杂的JD...【详细内容】
2023-12-08  Search: Netty  点击:(241)  评论:(0)  加入收藏
Netty入门实践-模拟IM聊天
我们使用的框架几乎都有网络通信的模块,比如常见的Dubbo、RocketMQ、ElasticSearch等。它们的网络通信模块使用Netty实现,之所以选择Netty,有2个主要原因: Netty封装了复杂的JDK...【详细内容】
2023-12-05  Search: Netty  点击:(108)  评论:(0)  加入收藏
如何使用Netty模拟一个Web服务端
Netty作为Web服务端具有以下好处:高性能Netty是一个基于事件驱动和异步非阻塞的网络编程框架,它使用了高效的NIO(非阻塞输入输出)模型。这使得Netty在处理大量并发连接时表现出...【详细内容】
2023-09-11  Search: Netty  点击:(272)  评论:(0)  加入收藏
玩转Netty,从“Hello World”开始!
为什么要用Netty?首先当然是NIO的使用,本身比较复杂,而且还存在一些问题。除此之外,如果在项目的开发中,要实现稳定的网络通信,就得考虑网络的闪断、客户端的重复接入、客户端的...【详细内容】
2023-05-16  Search: Netty  点击:(417)  评论:(0)  加入收藏
Netty和原生Java的性能比较
Netty是一种基于异步事件循环的网络应用编程方法。本文对比Netty与Java的本地服务器。虽然目前本地服务器使用的人不多,但我仍要找出netty比本地服务器的好处有多少。让我们...【详细内容】
2023-05-16  Search: Netty  点击:(88)  评论:(0)  加入收藏
基于Netty搭建WebSocket,模仿微信聊天页面
作者:小傅哥 博客:https://bugstack.cn 沉淀、分享、成长,让自己和他人都能有所收获!前言介绍 本章节我们模仿微信聊天页面,开发一个基于Netty搭建WebSocket通信案例。Netty的应...【详细内容】
2023-03-17  Search: Netty  点击:(237)  评论:(0)  加入收藏
Netty:遇到TCP发送缓冲区满了 写半包操作该如何处理
什么是写半包写半包:一份数据,一次发送没有把他全部发送,需要循环发送,那么第一次的操作称为写半包什么情况下会出现写半包:发送方发送200byte,但是接收方只能接受100byte,因此发送...【详细内容】
2023-03-13  Search: Netty  点击:(208)  评论:(0)  加入收藏
Java使用Netty框架自建DNS代理服务器教程
前言DNS协议作为着互联网客户端-服务器通信模式得第一关,在当下每天都有成千上亿上网记录产生得当今社会,其重要性自然不可言喻。在国内比较有名得DNS服务器有电信得114.114.1...【详细内容】
2023-02-09  Search: Netty  点击:(149)  评论:(0)  加入收藏
Netty基础介绍(使用场景、组件、模型、代码示例等)
Netty简介Netty 对 JDK 自带的 NIO 的 API 进行了良好的封装,解决了如客户端面临断线重连、 网络闪断、心跳处理、半包读写、 网络拥塞和异常流的处理等等问题。且Netty拥有...【详细内容】
2023-01-10  Search: Netty  点击:(146)  评论:(0)  加入收藏
从Redis、HTTP协议,看Nett协议设计,我发现了个惊天大秘密
1. 协议的作用TCP/IP 中消息传输基于流的方式,没有边界协议的目的就是划定消息的边界,制定通信双方要共同遵守的通信规则2. Redis 协议如果我们要向 Redis 服务器发送一条 set...【详细内容】
2023-01-03  Search: Netty  点击:(306)  评论:(0)  加入收藏
▌简易百科推荐
为什么Nginx被称为“反向”代理呢?
Nginx(发音为"engine-x")是一款高性能、轻量级的开源Web服务器软件,也可用作反向代理服务器、负载均衡器和HTTP缓存。Nginx之所以有被称为“反向”代理,是因为它充当客户端设备...【详细内容】
2024-02-01  coderidea  微信公众号  Tags:Nginx   点击:(60)  评论:(0)  加入收藏
哪种服务器操作系统更好呢?
在当今的IT世界中,服务器操作系统扮演着至关重要的角色。它们是确保服务器能够高效、安全地运行的关键因素。然而,对于许多人来说,服务器操作系统的种类和特点可能是一个复杂的...【详细内容】
2024-01-30    简易百科  Tags:操作系统   点击:(76)  评论:(0)  加入收藏
什么是VPS服务器
VPS服务器是一种虚拟化技术,它将一台物理服务器划分为多个虚拟的独立服务器,每个虚拟服务器都可以拥有自己的操作系统、运行环境、应用程序等。这种技术使得每个虚拟服务器可...【详细内容】
2024-01-30    简易百科  Tags:VPS服务器   点击:(70)  评论:(0)  加入收藏
VPS服务器下载速度慢?这五招帮你提速
VPS服务器下载速度慢可能会让用户感到沮丧,尤其是对于需要大量下载和上传数据的用户。幸运的是,有一些方法可以帮助您提高VPS服务器的下载速度,使您的在线体验更加顺畅。在本文...【详细内容】
2024-01-30  IDC行业观察者    Tags:VPS服务器   点击:(57)  评论:(0)  加入收藏
美国VPS和英国VPS:地理位置对服务器性能的影响
在今天的数字时代,VPS已成为在线业务和网站托管的关键组成部分。然而,选择合适的VPS主机服务时,地理位置通常被忽视,尽管它对服务器性能有着重要的影响。本文将探讨美国VPS和英...【详细内容】
2024-01-26  IDC行业观察者    Tags:服务器   点击:(55)  评论:(0)  加入收藏
如何判断服务器所需带宽:基于业务需求和流量模式的关键考量
在选择服务器时,带宽是一个重要的考虑因素。带宽的大小直接影响到网站的加载速度和用户的访问体验。那么,如何判断服务器需要多大的带宽呢?本文将为你揭示这一关键问题的答案...【详细内容】
2024-01-26  源库科技    Tags:服务器   点击:(74)  评论:(0)  加入收藏
服务器内存空间及IO操作原理解析
服务器的内存空间分为内核空间和用户空间,而我们编写的程序通常在用户空间中运行。在进行读写操作时,我们直接操作的是用户缓冲区,而用户缓冲区的内容来自于内核缓冲区。这种内...【详细内容】
2024-01-23  王建立    Tags:服务器   点击:(44)  评论:(0)  加入收藏
如何在Java环境中安装Nginx?
1. 下载Nginx:首先,前往Nginx官方网站(https://nginx.org/en/download.html)下载新版本的Nginx。选择适合您操作系统的版本,通常有Windows、Linux和Mac等不同操作系统的版本可供...【详细内容】
2024-01-22  敲代码的小动    Tags:Nginx   点击:(61)  评论:(0)  加入收藏
服务器证书和SSL证书有啥区别?
在互联网经济时代,随着越来越多的信息以及合作都是从企业官网开始的,因此绝大多数企业都会为自己的网站配置SSL证书,以提高安全性。在接触SSL证书时,也有很多人称之为服务器证书...【详细内容】
2024-01-10  安信SSL证书    Tags:服务器证书   点击:(65)  评论:(0)  加入收藏
宝塔面板怎样部署java项目?
宝塔面板怎样部署java项目?在使用宝塔面板部署Java项目之前,需要确保已经安装了Java Development Kit (JDK)。接下来,将介绍如何使用宝塔面板来部署Java项目的步骤。步骤一:安装...【详细内容】
2024-01-09  西部数码    Tags:宝塔面板   点击:(112)  评论:(0)  加入收藏
站内最新
站内热门
站内头条