一、前言

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

Netty的优点,简单一句话:使用简单、功能强大、性能强悍。

Netty的特点:

  • 高并发:Netty 是一款基于 NIO(Nonblocking IO,非阻塞IO)开发的网络通信框架,对比于 BIO(Blocking I/O,阻塞IO),他的并发性能得到了很大提高。
  • 传输快:Netty 的传输依赖于零拷贝特性,尽量减少不必要的内存拷贝,实现了更高效率的传输。
  • 封装好:Netty 封装了 NIO 操作的很多细节,提供了易于使用调用接口。

Netty的优势:

  • 使用简单:封装了 NIO 的很多细节,使用更简单。
  • 功能强大:预置了多种编解码功能,支持多种主流协议。
  • 定制能力强:可以通过 ChannelHandler 对通信框架进行灵活地扩展。
  • 性能高:通过与其他业界主流的 NIO 框架对比,Netty 的综合性能最优。
  • 稳定:Netty 修复了已经发现的所有 NIO 的 bug,让开发人员可以专注于业务本身。
  • 社区活跃:Netty 是活跃的开源项目,版本迭代周期短,bug 修复速度快。

Netty高性能表现在哪些方面?

  • IO 线程模型:同步非阻塞,用最少的资源做更多的事。
  • 内存零拷贝:尽量减少不必要的内存拷贝,实现了更高效率的传输。
  • 内存池设计:申请的内存可以重用,主要指直接内存。内部实现是用一颗二叉查找树管理内存分配情况。
  • 串形化处理读写:避免使用锁带来的性能开销。
  • 高性能序列化协议:支持 protobuf 等高性能序列化协议。

BIO、NIO和AIO的区别是什么?

这三个概念分别对应三种通讯模型:阻塞、非阻塞、非阻塞异步,概念这里就不写了,大家可以度娘搜一下,网上好多博客说Netty对应NIO,准确来说,应该是既可以是NIO,也可以是AIO,就看你怎么实现,这三个的区别如下:

  • BIO:一个连接一个线程,客户端有连接请求时服务器端就需要启动一个线程进行处理,线程开销大。伪异步IO:将请求连接放入线程池,一对多,但线程还是很宝贵的资源。
  • NIO:一个请求一个线程,但客户端发送的连接请求都会注册到多路复用器上,多路复用器轮询到连接有I/O请求时才启动一个线程进行处理。
  • AIO:一个有效请求一个线程,客户端的I/O请求都是由OS先完成了再通知服务器应用去启动线程进行处理。

BIO是面向流的,NIO是面向缓冲区的;BIO的各种流是阻塞的。而NIO是非阻塞的;BIO的Stream是单向的,而NIO的channel是双向的。

NIO的特点:事件驱动模型、单线程处理多任务、非阻塞I/O,I/O读写不再阻塞,而是返回0、基于block的传输比基于流的传输更高效、更高级的IO函数zero-copy、IO多路复用大大提高了Java网络应用的可伸缩性和实用性。基于Reactor线程模型。

二、Netty能做什么?

学技能都是为了能够应用到实际工作中去,谁也不是为了学而学、弄着玩不是,那么Netty能做什么呢?主要是在两个方面:

现在物联网的应用无处不在,大量的项目都牵涉到应用传感器和服务器端的数据通信,Netty作为基础通信组件、能够轻松解决之前有较高门槛的通信系统开发,你不用再为如何解析各类简单、或复杂的通讯协议而薅头发了,有过这方面开发经验的程序员会有更深刻、或者说刻骨铭心的体会。

现在互联网系统讲究的都是高并发、分布式、微服务,各类消息满天飞,Netty在这类架构里面的应用可谓是如鱼得水,如果你对当前的各种应用服务器不爽,那么完全可以基于Netty来实现自己的HTTP服务器,FTP服务器,UDP服务器,RPC服务器,WebSocket服务器,Redis的Proxy服务器,MySQL的Proxy服务器等等。

三、掌握Netty有什么好处呢?

直接的好处是:能够有进大厂、拿高薪的机会,业内好多著名的公司在招聘高级/资深Java工程师时基本上都要求熟练掌握、或熟悉Netty。

这个名单还可以很长很长。。。

作为一个学Java的,如果没有研究过Netty,那么你对Java语言的使用和理解仅仅停留在表面水平,会点SSH,写几个MVC,访问数据库和缓存,这些只是初、中等Java程序员干的事。如果你要进阶,想了解Java服务器的深层高阶知识,Netty绝对是一个必须要过的门槛。

间接地好处是:多款开源框架中应用了Netty,掌握了Netty,就具有分析这些开源框架的基础了,也就是有了成为技术大牛的基础。

这些开源框架有哪些呢?简单罗列一些典型的,如下:

  • 阿里分布式服务框架 Dubbo 的 RPC 框架;
  • 淘宝的消息中间件 RocketMQ;
  • Hadoop 的高性能通信和序列化组件 Avro 的 RPC 框架;
  • 开源集群运算框架 Spark;
  • 分布式计算框架 Storm;
  • 并发应用和分布式应用 Akka;
  • 名单依然很长很长。。。。

四、动手开干、实现一个传输字符串的简单实例

在开始动手之前,必要的基础概念还是要知道的,要不然代码敲下来,功能倒是实现了,但对Netty还是一头雾水,这就不是本文要达到的目的了。

本示例需要用到的基础知识主要有以下几方面的东东,这些知识点最好有一个大概的了解,要不然,看实例会有一定的困难。

  • 掌握Java基础
  • 掌握Maven基础
  • 熟悉IntelliJ IDEA集成开发工具的使用,这个工具简称IDEA
  • 知道TCP、Socket的基本概念

1、Netty的组件

  • I/O:各种各样的流(文件、数组、缓冲、管道。。。)的处理(输入输出)。
  • Channel:通道,代表一个连接,每个Client请对会对应到具体的一个Channel。
  • ChannelPipeline:责任链,每个Channel都有且仅有一个ChannelPipeline与之对应,里面是各种各样的Handler。
  • handler:用于处理出入站消息及相应的事件,实现我们自己要的业务逻辑。
  • EventLoopGroup:I/O线程池,负责处理Channel对应的I/O事件。
  • ServerBootstrap:服务器端启动辅助对象。
  • Bootstrap:客户端启动辅助对象。
  • ChannelInitializer:Channel初始化器。
  • ChannelFuture:代表I/O操作的执行结果,通过事件机制,获取执行结果,通过添加监听器,执行我们想要的操作。
  • ByteBuf:字节序列,通过ByteBuf操作基础的字节数组和缓冲区。

2、基础环境准备

基础环境准备主要有三个方面:JDK安装及环境变量设置、Maven安装及环境变量设置、IDEA安装及基本设置。

2.1、JDK安装及环境变量设置

JDK下载,可以从官方现在,也可以度娘上随便搜下载链接,最新版是JDK14,我这里下载的是JDK8,用8还是14哪个版本无所谓,都可以,但要注意一点的是,现在从JDK的官网Oracle下载需要账号了,没账号的可下不了啦,不知道在搞什么东东。

官网下载地址:https://www.oracle.com ,截图如下:

下载完,一路Next安装完,在创建Java环境变量设置,[此电脑]右键-->[属性]-->[高级系统设置]-->[环境变量]-->[系统变量],截图如下:

Java环境变量创建完毕后,在DOS窗口执行命令:java -version,测试一下是否正常

2.2、Maven安装及环境变量设置

Maven功能很强大,但大家不用担心、本实例中仅仅是利用其便利的jar包依赖、jar包依赖传递,基本上没有任何学习成本。

jar包依赖、jar包依赖传递的概念如下图,清楚明了,都不用多做解释:

Maven是下载,解压缩后,配置环境变量后就能用,不用安装的。

下载地址https://downloads.apache.org/maven/maven-3/3.6.3/binaries/

安装:下载压缩包,解压,文件夹拷贝到所想存储的位置(如C盘根目录)

配置环境变量,和Java的环境变量配置一样的,创建MAVEN_HOME,指向Maven文件夹,再在path中添加进去就行,如下图:

由于直接冲Maven的中央仓库中自动下载jar包较慢,一般在Maven的配置文件中,增加阿里云的公共仓库配置,这样会显著加快jar包的下载速度,如下:

上面的环境变量设置完后,通过DOS窗口中输入命令:mvn -version 进行验证是否成功,如下:

2.3、IDEA安装及基本设置

IDEA的下载和安装就不多说了,其版本分旗舰版和社区版,旗舰版收费,社区版免费,社区版不支持html、js、css等,但对于本实例,社区版就够用了,但如果你不在乎那点银子,可以考虑旗舰版,一步到位,万一后面我们还要做WEB系统开发可以免得折腾。

其安装不用多说,一路Next就行,安装完后,在其配置里面指定一下JDK、Maven的位置就行了,如下图:Maven指定:[File]-->[setting]-->[Build,Excution,Deployment]-->[Build Tools]-->[Maven]

JDK指定:[File]-->[Project Structure]-->[Project Setting]-->[Project]

3、在IDEA中创建Maven工程

新建工程

填写包名及工程名称

Maven配置

生成工程,自动创建Maven的依赖文件

在pom.xml中配置Netty依赖

经过上面的步骤,我们的Maven工程就已经创建完毕,现在可以编写Netty的第一个程序,这个程序很简单,传输一个字符串,虽然程序很简单,但是已经能够大体上反映Netty开发通信程序的一个整体流程了。

4、Netty开发的基本流程

Netty开发的基本流程很简洁,服务器端和客户端都是这个套路,如下:

Netty开发的实际过程,这是一个简化的过程,但已经把大概流程表达出来了,绿色的代表客户端流程、蓝色的代表服务器端流程,注意标红的部分,见下图:

4.1 创建客户端类

创建Handler

首先创建Handler类,该类用于接收服务器端发送的数据,这是一个简化的类,只重写了消息读取方法channelRead0、捕捉异常方法exceptionCaught。

客户端的Handler一般继承的是SimpleChannelInboundHandler,该类有丰富的方法,心跳、超时检测、连接状态等等。

代码如下:

package com.jcj.helloworld;import io.netty.buffer.ByteBuf;import io.netty.channel.ChannelHandler;import io.netty.channel.ChannelHandlerContext;import io.netty.channel.SimpleChannelInboundHandler;import io.netty.util.CharsetUtil;/** * @Auther:程序员麦冬 * @Date: 2020/8/4 15:25 * @Description: 通用handler,处理I/O事件 */@ChannelHandler.Sharablepublic class HandlerClientHello extends SimpleChannelInboundHandler{    @Override    protected void channelRead0(ChannelHandlerContext channelHandlerContext, ByteBuf byteBuf) throws Exception    {        /**        * @Author 程序员麦冬        * @Date 2020/8/4 15:27        * @Description  处理接收到的消息        **/        System.out.println("接收到的消息:"+byteBuf.toString(CharsetUtil.UTF_8));    }    @Override    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception    {        /**        * @Author 程序员麦冬        * @Date 2020/8/4 15:30        * @Description  处理I/O事件的异常        **/        cause.printStackTrace();        ctx.close();    }}

代码说明:

  • @ChannelHandler.Sharable,这个注解是为了线程安全,如果你不在乎是否线程安全,不加也可以。
  • SimpleChannelInboundHandler,这里的类型可以是ByteBuf,也可以是String,还可以是对象,根据实际情况来。
  • channelRead0,消息读取方法,注意名称中有个0。
  • ChannelHandlerContext,通道上下文,代指Channel。
  • ByteBuf,字节序列,通过ByteBuf操作基础的字节数组和缓冲区,因为JDK原生操作字节麻烦、效率低,所以Netty对字节的操作进行了封装,实现了指数级的性能提升,同时使用更加便利。
  • CharsetUtil.UTF_8,这个是JDK原生的方法,用于指定字节数组转换为字符串时的编码格式。

创建客户端启动类

客户端启动类根据服务器端的IP和端口,建立连接,连接建立后,实现消息的双向传输。

代码较简洁,如下:

package com.jcj.helloworld;import com.sun.org.apache.bcel.internal.generic.ATHROW;import io.netty.bootstrap.Bootstrap;import io.netty.buffer.Unpooled;import io.netty.channel.ChannelFuture;import io.netty.channel.ChannelInitializer;import io.netty.channel.EventLoopGroup;import io.netty.channel.nio.NioEventLoopGroup;import io.netty.channel.socket.SocketChannel;import io.netty.channel.socket.nio.NioSocketChannel;import io.netty.util.CharsetUtil;import java.net.InetSocketAddress;/** * @Auther:  * @Date: 2020/6/1 11:24 * @Description: 客户端启动类 */public class AppClientHello{    private final String host;    private final int port;    public AppClientHello(String host, int port)    {        this.host = host;        this.port = port;    }    public void run() throws Exception    {        /**        * @Author 江成军        * @Date 2020/6/1 11:28        * @Description  配置相应的参数,提供连接到远端的方法        **/        EventLoopGroup group = new NioEventLoopGroup();//I/O线程池        try {            Bootstrap bs = new Bootstrap();//客户端辅助启动类            bs.group(group)                    .channel(NioSocketChannel.class)//实例化一个Channel                    .remoteAddress(new InetSocketAddress(host,port))                    .handler(new ChannelInitializer()//进行通道初始化配置                    {                        @Override                        protected void initChannel(SocketChannel socketChannel) throws Exception                        {                            socketChannel.pipeline().addLast(new HandlerClientHello());//添加我们自定义的Handler                        }                    });            //连接到远程节点;等待连接完成            ChannelFuture future=bs.connect().sync();            //发送消息到服务器端,编码格式是utf-8            future.channel().writeAndFlush(Unpooled.copiedBuffer("Hello World", CharsetUtil.UTF_8));            //阻塞操作,closeFuture()开启了一个channel的监听器(这期间channel在进行各项工作),直到链路断开            future.channel().closeFuture().sync();        } finally {            group.shutdownGracefully().sync();        }    }    public static void main(String[] args) throws Exception    {        new AppClientHello("127.0.0.1",18080).run();    }}

由于代码中已经添加了详尽的注释,这里只对极个别的进行说明:

  • ChannelInitializer,通道Channel的初始化工作,如加入多个handler,都在这里进行。
  • bs.connect().sync(),这里的sync()表示采用的同步方法,这样连接建立成功后,才继续往下执行。
  • pipeline(),连接建立后,都会自动创建一个管道pipeline,这个管道也被称为责任链,保证顺序执行,同时又可以灵活的配置各类Handler,这是一个很精妙的设计,既减少了线程切换带来的资源开销、避免好多麻烦事,同时性能又得到了极大增强。

4.2 创建服务器端类

创建Handler

和客户端一样,只重写了消息读取方法channelRead(注意这里不是channelRead0)、捕捉异常方法exceptionCaught。

另外服务器端Handler继承的是ChannelInboundHandlerAdapter,而不是SimpleChannelInboundHandler,至于这两者的区别,这里不赘述,大家自行百度吧。

代码如下:

package com.jcj.helloworld;import io.netty.buffer.ByteBuf;import io.netty.buffer.Unpooled;import io.netty.channel.ChannelHandler;import io.netty.channel.ChannelHandlerContext;import io.netty.channel.ChannelInboundHandlerAdapter;import io.netty.util.CharsetUtil;/** * @Auther: 程序员麦冬 * @Date: 2020/8/4 11:47 * @Description: 服务器端I/O处理类 */@ChannelHandler.Sharablepublic class HandlerServerHello extends ChannelInboundHandlerAdapter{    @Override    public void channelRead(ChannelHandlerContext ctx, Object msg)  throws Exception    {        //处理收到的数据,并反馈消息到到客户端        ByteBuf in = (ByteBuf) msg;        System.out.println("收到客户端发过来的消息: " + in.toString(CharsetUtil.UTF_8));        //写入并发送信息到远端(客户端)        ctx.writeAndFlush(Unpooled.copiedBuffer("你好,我是服务端,我已经收到你发送的消息", CharsetUtil.UTF_8));    }    @Override    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception    {        //出现异常的时候执行的动作(打印并关闭通道)        cause.printStackTrace();        ctx.close();    }}

以上代码很简洁,大家注意和客户端Handler类进行比较。

创建服务器端启动类

服务器端启动类比客户端启动类稍显复杂一点,先贴出代码如下:

package com.jcj.helloworld;import io.netty.bootstrap.ServerBootstrap;import io.netty.channel.ChannelFuture;import io.netty.channel.ChannelInitializer;import io.netty.channel.EventLoopGroup;import io.netty.channel.nio.NioEventLoopGroup;import io.netty.channel.socket.SocketChannel;import io.netty.channel.socket.nio.NioServerSocketChannel;import java.net.InetSocketAddress;/** * @Auther: 程序员麦冬 * @Date: 2020/8/4 11:51 * @Description: 服务器端启动类 */public class AppServerHello{    private int port;    public AppServerHello(int port)    {        this.port = port;    }    public void run() throws Exception    {        EventLoopGroup group = new NioEventLoopGroup();//Netty的Reactor线程池,初始化了一个NioEventLoop数组,用来处理I/O操作,如接受新的连接和读/写数据        try {            ServerBootstrap b = new ServerBootstrap();//用于启动NIO服务            b.group(group)                    .channel(NioServerSocketChannel.class) //通过工厂方法设计模式实例化一个channel                    .localAddress(new InetSocketAddress(port))//设置监听端口                    .childHandler(new ChannelInitializer() {                        //ChannelInitializer是一个特殊的处理类,他的目的是帮助使用者配置一个新的Channel,用于把许多自定义的处理类增加到pipline上来                        @Override                        public void initChannel(SocketChannel ch) throws Exception {//ChannelInitializer 是一个特殊的处理类,他的目的是帮助使用者配置一个新的 Channel。                            ch.pipeline().addLast(new HandlerServerHello());//配置childHandler来通知一个关于消息处理的InfoServerHandler实例                        }                    });            //绑定服务器,该实例将提供有关IO操作的结果或状态的信息            ChannelFuture channelFuture= b.bind().sync();            System.out.println("在" + channelFuture.channel().localAddress()+"上开启监听");            //阻塞操作,closeFuture()开启了一个channel的监听器(这期间channel在进行各项工作),直到链路断开            channelFuture.channel().closeFuture().sync();        } finally {            group.shutdownGracefully().sync();//关闭EventLoopGroup并释放所有资源,包括所有创建的线程        }    }    public static void main(String[] args)  throws Exception    {        new AppServerHello(18080).run();    }}

代码说明:

  • EventLoopGroup,实际项目中,这里创建两个EventLoopGroup的实例,一个负责接收客户端的连接,另一个负责处理消息I/O,这里为了简单展示流程,让一个实例把这两方面的活都干了。
  • NioServerSocketChannel,通过工厂通过工厂方法设计模式实例化一个channel,这个在大家还没有能够熟练使用Netty进行项目开发的情况下,不用去深究。

到这里,我们就把服务器端和客户端都写完了 ,如何运行呢,先在服务器端启动类上右键,点Run 'AppServerHello.main()'菜单运行,见下图:

然后,再同样的操作,运行客户端启动类,就能看见效果了。

5、尾声

本文的内容就到这里结束了,希望本文能够让大家对Netty有一个整体的认识,并大概了解其开发流程。

关注我,私信回复【资料】即可领取视频中java相关资料以及一份227页最新的bat大厂面试宝典

欢迎大家一起交流,喜欢文章记得关注我点个赞哟,感谢支持!

maven netty 配置_进阿里、腾讯、字节跳动、美团必掌握的Netty相关推荐

  1. 互联网大厂Java岗考点(阿里+百度+腾讯+字节跳动+美团+京东)

    本文扼要 本文结构主要分为以下三个部分: 01 互联网大厂考点(阿里+百度+腾讯+字节跳动+美团+京东) 02 Java 面试考点大全(基本功底+常用技术+技术深度+技术经验+学习能力+工作能力+项目 ...

  2. 互联网大厂考点(阿里+百度+腾讯+字节跳动+美团+京东)

    本文扼要 本文结构主要分为以下三个部分: 01 互联网大厂考点(阿里+百度+腾讯+字节跳动+美团+京东) 02 Java 面试考点大全(基本功底+常用技术+技术深度+技术经验+学习能力+工作能力+项目 ...

  3. 2022前端秋招面试题总结 阿里 腾讯 字节 百度 网易 京东 小红书 快手面试记录

    2022前端秋招面试题总结 阿里 腾讯 字节 百度 网易 京东 小红书 快手面试记录 等了百度三个月,终于发offer了,白菜价,92大佬们拒的薪资,我知足了. 排序挂了狠多,快手,蚂蚁,- 很绝望 ...

  4. maven netty 配置_Netty是业界最流行的NIO框架之一:初识Netty

    为什么使用Netty Netty是业界最流行的NIO框架之一,它的健壮性.功能.性能.可定制性.可扩展性在同类框架中都是首屈一指的,它已经得到了成百上千的商用项目的证明.对于为什么使用Netty这个话 ...

  5. 2020年9家互联网大厂职级和薪酬曝光!含阿里腾讯字节美团……

    文章来源:运营黑客社区(ID:yunyingheike) 作者:Vivi 以 BAT 为代表的互联网大厂,一直是求职者眼中的香饽饽,"大厂经历"在国内就业环境中无异于一块金子招牌. ...

  6. 阿里腾讯字节,谁才是真正的福利大厂

    最近恰逢金三银四,是跳槽求职的好机会,不少人都在这个时候找工作,小编看到过不少校招或者社招的薪水汇总,但实习岗的还是比较少见. 今天我们就来罗列一下字节跳动.腾讯.拼多多.阿里.美团实习岗薪水及福利待 ...

  7. netty依赖_高性能:《一遍文章带你看懂 Netty世界》

    [概述] Netty是由JBOSS提供的基于Java NIO的开源框架,Netty提供异步非阻塞.事件驱动.高性能.高可靠.高可定制性的网络应用程序和工具,可用于开发服务端和客户端. 一. Netty ...

  8. leetcode 滴滴_拿到腾讯字节快手offer后,他的LeetCode刷题经验GitHub获1300星

    杨净 发自 凹非寺 量子位 报道 | 公众号 QbitAI 今天,你刷LeetCode了吗? 最近,GitHub上这样一套LeetCode笔记火了. 一位Java研发工程师分享了一个名为「LeetCo ...

  9. 秋招腾讯字节跳动小米前端面经

    欢迎阅读 作者的个人网站: Greatiga的个人网站.会不定时分享各种学习心得文章 前言 目前已经毕业,在去年2020秋招的摸爬滚打中,侥幸收获了几个小offer.最后也很幸运去腾讯实习了一段时间. ...

最新文章

  1. 基于MATLAB的面向对象编程(4)——类文件
  2. ASP.NET页面指令
  3. 数据结构——HDU1312:Red and Black(DFS)
  4. 银行存取款系统的设计与实现c语言,【银行|银行存取款管理系统设计】取款|管理|系统|设计-傻大方...
  5. 【Java多线程编程】选号程序
  6. python列表切片赋值_Python中对切片赋值原理分析
  7. 原创电子书:C#难点逐个击破
  8. tomcat服务自动关闭_windows10系统关闭自动更新服务
  9. dhcp snooping华为_使用DHCP snooping 功能防止DHCP Server仿冒者攻击(华为交换机)
  10. SaltStack 模块学习之拷贝master服务器上文件和目录到minion服务器
  11. 最大子数列和的问题求解
  12. Ubuntu命令大全(转载)
  13. php url编码解码
  14. Android 用代码Ping网络
  15. winform backgroundWorker 用法
  16. 史上最全的数据链路层基础知识详解
  17. c语言 音符符号大全,音符符号大全
  18. linux内核去掉pty,请问如何升级内核?高手请进!!!
  19. 支付宝扫福最佳攻略,你值得拥有
  20. ftp服务器型号,ftp服务器的类型及其特点

热门文章

  1. 四边形可以分为几类_展台设计搭建可以分为哪几类?
  2. 时尚高端电商网页设计素材
  3. 梦幻飞屋创意房产海报,温馨画面,梦中的家园
  4. pr调色预设_视频调色不好掌握?用这2000套PR、AE、达芬奇调色预设吧
  5. Linux内核网络协议栈:udp数据包发送(源码解读)
  6. Linux网络协议指令:ifconfig/netstat(net-tools)工具 .vs. iproute2
  7. Seastar:多核机器上编写高效复杂的服务器应用程序的 C++ 库
  8. C语言curl实现FTP上传、下载、获取文件信息
  9. INET Sockets Tour: ioctl()
  10. linux平台下C语言按进程名查找进程号pid