目录
NIO
Reactor线程模型
总结
为什么选择Netty
什么是TCP 粘包/拆包
透过现象分析原因
如何解决
Netty 的零拷贝
传统意义的拷贝
零拷贝的概念
Netty中的零拷贝
Netty 内部执行流程
首页 Java java教程 Netty是什么?Netty相关知识的深入解析

Netty是什么?Netty相关知识的深入解析

Nov 24, 2018 pm 04:22 PM
netty

本篇文章给大家带来的内容是关于Netty是什么?Netty相关知识的深入解析,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。

Netty到底是什么

从HTTP说起

有了Netty,你可以实现自己的HTTP服务器,FTP服务器,UDP服务器,RPC服务器,WebSocket服务器,Redis的Proxy服务器,MySQL的Proxy服务器等等。

我们回顾一下传统的HTTP服务器的原理

1、创建一个ServerSocket,监听并绑定一个端口

2、一系列客户端来请求这个端口

3、服务器使用Accept,获得一个来自客户端的Socket连接对象

4、启动一个新线程处理连接

4.1、读Socket,得到字节流

4.2、解码协议,得到Http请求对象

4.3、处理Http请求,得到一个结果,封装成一个HttpResponse对象

4.4、编码协议,将结果序列化字节流 写Socket,将字节流发给客户端

5、继续循环步骤3

HTTP服务器之所以称为HTTP服务器,是因为编码解码协议是HTTP协议,如果协议是Redis协议,那它就成了Redis服务器,如果协议是WebSocket,那它就成了WebSocket服务器,等等。 使用Netty你就可以定制编解码协议,实现自己的特定协议的服务器。

NIO

上面是一个传统处理http的服务器,但是在高并发的环境下,线程数量会比较多,System load也会比较高,于是就有了NIO。

他并不是Java独有的概念,NIO代表的一个词汇叫着IO多路复用。它是由操作系统提供的系统调用,早期这个操作系统调用的名字是select,但是性能低下,后来渐渐演化成了Linux下的epoll和Mac里的kqueue。我们一般就说是epoll,因为没有人拿苹果电脑作为服务器使用对外提供服务。而Netty就是基于Java NIO技术封装的一套框架。为什么要封装,因为原生的Java NIO使用起来没那么方便,而且还有臭名昭著的bug,Netty把它封装之后,提供了一个易于操作的使用模式和接口,用户使用起来也就便捷多了。

说NIO之前先说一下BIO(Blocking IO),如何理解这个Blocking呢?

客户端监听(Listen)时,Accept是阻塞的,只有新连接来了,Accept才会返回,主线程才能继

读写socket时,Read是阻塞的,只有请求消息来了,Read才能返回,子线程才能继续处理

读写socket时,Write是阻塞的,只有客户端把消息收了,Write才能返回,子线程才能继续读取下一个请求

传统的BIO模式下,从头到尾的所有线程都是阻塞的,这些线程就干等着,占用系统的资源,什么事也不干。

那么NIO是怎么做到非阻塞的呢。它用的是事件机制。它可以用一个线程把Accept,读写操作,请求处理的逻辑全干了。如果什么事都没得做,它也不会死循环,它会将线程休眠起来,直到下一个事件来了再继续干活,这样的一个线程称之为NIO线程。用伪代码表示:

while true {

    events = takeEvents(fds)  // 获取事件,如果没有事件,线程就休眠

    for event in events {        if event.isAcceptable {

            doAccept() // 新链接来了
        } elif event.isReadable {

            request = doRead() // 读消息

            if request.isComplete() {

                doProcess()

            }

        } elif event.isWriteable {

            doWrite()  // 写消息
        }

    }

}
登录后复制

Reactor线程模型

Reactor单线程模型

一个NIO线程 一个accept线程:

Netty是什么?Netty相关知识的深入解析

Reactor多线程模型

Netty是什么?Netty相关知识的深入解析

Reactor主从模型

主从Reactor多线程:多个acceptor的NIO线程池用于接受客户端的连接

Netty是什么?Netty相关知识的深入解析

Netty可以基于如上三种模型进行灵活的配置。

总结

Netty是建立在NIO基础之上,Netty在NIO之上又提供了更高层次的抽象。

在Netty里面,Accept连接可以使用单独的线程池去处理,读写操作又是另外的线程池来处理。

Accept连接和读写操作也可以使用同一个线程池来进行处理。而请求处理逻辑既可以使用单独的线程池进行处理,也可以跟放在读写线程一块处理。线程池中的每一个线程都是NIO线程。用户可以根据实际情况进行组装,构造出满足系统需求的高性能并发模型。

为什么选择Netty

如果不用netty,使用原生JDK的话,有如下问题:

1、API复杂

2、对多线程很熟悉:因为NIO涉及到Reactor模式

3、高可用的话:需要出路断连重连、半包读写、失败缓存等问题

4、JDK NIO的bug

而Netty来说,他的api简单、性能高而且社区活跃(dubbo、rocketmq等都使用了它)

什么是TCP 粘包/拆包

现象

先看如下代码,这个代码是使用netty在client端重复写100次数据给server端,ByteBuf是netty的一个字节容器,里面存放是的需要发送的数据

public class FirstClientHandler extends ChannelInboundHandlerAdapter { 
 @Override 
 public void channelActive(ChannelHandlerContext ctx) { 
 for (int i = 0; i < 1000; i++) { 
 ByteBuf buffer = getByteBuf(ctx); 
 ctx.channel().writeAndFlush(buffer); 
 } 
 } 
 private ByteBuf getByteBuf(ChannelHandlerContext ctx) { 
 byte[] bytes = "需要更多资料加群:586446657".getBytes(Charset.forName("utf-8")); 
 ByteBuf buffer = ctx.alloc().buffer(); 
 buffer.writeBytes(bytes); 
 return buffer; 
 }
}
登录后复制

从client端读取到的数据为:

Netty是什么?Netty相关知识的深入解析

从服务端的控制台输出可以看出,存在三种类型的输出

一种是正常的字符串输出。

一种是多个字符串“粘”在了一起,我们定义这种 ByteBuf 为粘包。

一种是一个字符串被“拆”开,形成一个破碎的包,我们定义这种 ByteBuf 为半包。

透过现象分析原因

应用层面使用了Netty,但是对于操作系统来说,只认TCP协议,尽管我们的应用层是按照 ByteBuf 为 单位来发送数据,server按照Bytebuf读取,但是到了底层操作系统仍然是按照字节流发送数据,因此,数据到了服务端,也是按照字节流的方式读入,然后到了 Netty 应用层面,重新拼装成 ByteBuf,而这里的 ByteBuf 与客户端按顺序发送的 ByteBuf 可能是不对等的。因此,我们需要在客户端根据自定义协议来组装我们应用层的数据包,然后在服务端根据我们的应用层的协议来组装数据包,这个过程通常在服务端称为拆包,而在客户端称为粘包。

拆包和粘包是相对的,一端粘了包,另外一端就需要将粘过的包拆开,发送端将三个数据包粘成两个 TCP 数据包发送到接收端,接收端就需要根据应用协议将两个数据包重新组装成三个数据包。

如何解决

在没有 Netty 的情况下,用户如果自己需要拆包,基本原理就是不断从 TCP 缓冲区中读取数据,每次读取完都需要判断是否是一个完整的数据包 如果当前读取的数据不足以拼接成一个完整的业务数据包,那就保留该数据,继续从 TCP 缓冲区中读取,直到得到一个完整的数据包。 如果当前读到的数据加上已经读取的数据足够拼接成一个数据包,那就将已经读取的数据拼接上本次读取的数据,构成一个完整的业务数据包传递到业务逻辑,多余的数据仍然保留,以便和下次读到的数据尝试拼接。

而在Netty中,已经造好了许多类型的拆包器,我们直接用就好:

Netty是什么?Netty相关知识的深入解析

选好拆包器后,在代码中client段和server端将拆包器加入到chanelPipeline之中就好了:

如上实例中:

客户端:

ch.pipeline().addLast(new FixedLengthFrameDecoder(31));
登录后复制
登录后复制

服务端:

ch.pipeline().addLast(new FixedLengthFrameDecoder(31));
登录后复制
登录后复制

Netty是什么?Netty相关知识的深入解析

Netty 的零拷贝

传统意义的拷贝

是在发送数据的时候,传统的实现方式是:

1. `File.read(bytes)`

2. `Socket.send(bytes)`

这种方式需要四次数据拷贝和四次上下文切换:

1. 数据从磁盘读取到内核的read buffer

2. 数据从内核缓冲区拷贝到用户缓冲区

3. 数据从用户缓冲区拷贝到内核的socket buffer

4. 数据从内核的socket buffer拷贝到网卡接口(硬件)的缓冲区

零拷贝的概念

明显上面的第二步和第三步是没有必要的,通过java的FileChannel.transferTo方法,可以避免上面两次多余的拷贝(当然这需要底层操作系统支持)

1. 调用transferTo,数据从文件由DMA引擎拷贝到内核read buffer

2. 接着DMA从内核read buffer将数据拷贝到网卡接口buffer

上面的两次操作都不需要CPU参与,所以就达到了零拷贝。

Netty中的零拷贝

主要体现在三个方面:

1、bytebuffer

Netty发送和接收消息主要使用bytebuffer,bytebuffer使用对外内存(DirectMemory)直接进行Socket读写。

原因:如果使用传统的堆内存进行Socket读写,JVM会将堆内存buffer拷贝一份到直接内存中然后再写入socket,多了一次缓冲区的内存拷贝。DirectMemory中可以直接通过DMA发送到网卡接口

2、Composite Buffers

传统的ByteBuffer,如果需要将两个ByteBuffer中的数据组合到一起,我们需要首先创建一个size=size1 size2大小的新的数组,然后将两个数组中的数据拷贝到新的数组中。但是使用Netty提供的组合ByteBuf,就可以避免这样的操作,因为CompositeByteBuf并没有真正将多个Buffer组合起来,而是保存了它们的引用,从而避免了数据的拷贝,实现了零拷贝。

3、对于FileChannel.transferTo的使用

Netty中使用了FileChannel的transferTo方法,该方法依赖于操作系统实现零拷贝。

Netty 内部执行流程

服务端:

 Netty是什么?Netty相关知识的深入解析

Netty是什么?Netty相关知识的深入解析

1、创建ServerBootStrap实例

2、设置并绑定Reactor线程池:EventLoopGroup,EventLoop就是处理所有注册到本线程的Selector上面的Channel

3、设置并绑定服务端的channel

4、5、创建处理网络事件的ChannelPipeline和handler,网络时间以流的形式在其中流转,handler完成多数的功能定制:比如编解码 SSl安全认证

6、绑定并启动监听端口

7、当轮训到准备就绪的channel后,由Reactor线程:NioEventLoop执行pipline中的方法,最终调度并执行channelHandler 

8、说到这里顺便给大家推荐一个Java的交流学习社区:586446657,里面不仅可以交流讨论,还有面试经验分享以及免费的资料下载,包括Spring,MyBatis,Netty源码分析,高并发、高性能、分布式、微服务架构的原理,JVM性能优化这些成为架构师必备的知识体系。相信对于已经工作和遇到技术瓶颈的码友,在这里会有你需要的内容。

客户端

Netty是什么?Netty相关知识的深入解析

Netty是什么?Netty相关知识的深入解析

总结

以上就是我对Netty相关知识整理,如果有不同的见解,欢迎讨论!

以上是Netty是什么?Netty相关知识的深入解析的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
4 周前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳图形设置
4 周前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您听不到任何人,如何修复音频
1 个月前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.聊天命令以及如何使用它们
1 个月前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

Java API 开发中使用 Netty4 进行 TCP 通信 Java API 开发中使用 Netty4 进行 TCP 通信 Jun 17, 2023 pm 11:18 PM

TCP是计算机网络通信协议的一种,是一种面向连接的传输协议。在Java应用开发中,TCP通信被广泛应用于各种场景,比如客户端和服务器之间的数据传输、音视频实时传输等等。Netty4是一个高性能、高可扩展性、高性能的网络编程框架,能够优化服务器和客户端之间的数据交换过程,使其更加高效可靠。使用Netty4进行TCP通信的具体实现步骤如下:引入

Java开发:如何使用Netty进行高性能网络编程 Java开发:如何使用Netty进行高性能网络编程 Sep 20, 2023 pm 02:09 PM

Java开发:如何使用Netty进行高性能网络编程摘要:Netty是一个高性能、异步事件驱动的网络编程框架,能够简化网络应用程序的开发过程。本文将介绍Netty的主要特点以及如何使用Netty进行高性能网络编程。同时,我们还会提供一些具体的Java代码示例,帮助读者更好地理解和应用Netty。一、Netty简介Netty是一个基于JavaNIO的网络编程框

PHP实现开源Netty框架 PHP实现开源Netty框架 Jun 18, 2023 pm 07:03 PM

随着互联网技术的不断发展,网络程序设计变得越来越重要。在这个领域中,Netty是一个十分知名的框架。它是一个高性能、异步事件驱动的网络应用程序框架,被广泛用于开发各种高并发的网络应用程序。Netty是一个Java框架,它的产生推动了Java网络编程的发展。然而,随着PHP的广泛使用,PHP开发者也在寻找能够胜任高并发网络编程的框架。因此,本文介绍了如何利用P

springboot整合netty框架的方式有哪些 springboot整合netty框架的方式有哪些 May 10, 2023 pm 09:55 PM

netty作为一个高性能的io框架,是非好用的一个技术框架,Netty是一个基于NIO的客户、服务器端编程框架,使用Netty可以确保你快速和简单的开发出一个网络应用,例如实现了某种协议的客户、服务端应用。Netty相当于简化和流线化了网络应用的编程开发过程,例如:基于TCP和UDP的socket服务开发。“快速”和“简单”并不用产生维护性或性能上的问题。Netty是一个吸收了多种协议(包括FTP、SMTP、HTTP等各种二进制文本协议)的实现经验,并经过相当精心设计的项目。最终,Netty成功

Java后端开发:使用Netty构建高并发API服务器 Java后端开发:使用Netty构建高并发API服务器 Jun 17, 2023 am 10:23 AM

随着互联网的不断发展和应用领域的不断扩展,高并发成为了网络应用开发中必须考虑的问题,而Java作为一种广泛应用于企业级应用开发的语言,其在高并发应用场景下的表现备受关注。Netty是一款高性能、异步事件驱动的网络应用框架,近年来在Java后端开发领域享有广泛的应用。本文将介绍Netty的基本概念和使用方法,并以搭建一个高并发的API服务器为例,展示Netty

如何使用Java开发一个基于Netty的高性能网络应用 如何使用Java开发一个基于Netty的高性能网络应用 Sep 20, 2023 pm 12:21 PM

如何使用Java开发一个基于Netty的高性能网络应用Netty是一种基于JavaNIO技术的网络编程框架,被广泛应用于高性能的网络应用开发。在本文中,我们将探讨如何使用Java和Netty来开发一个基于Netty的高性能网络应用。我们将介绍Netty的基本概念和特性,并提供一些代码示例以帮助你更好地理解和使用Netty。一、Netty的基本概念和特性Ne

Java API 开发中使用 Netty5 进行 TCP 通信 Java API 开发中使用 Netty5 进行 TCP 通信 Jun 18, 2023 am 08:31 AM

在JavaAPI开发中,TCP通信是一个非常重要的组件,而Netty5是一套基于NIO的高性能网络通信框架,可以非常方便地处理复杂的网络通信任务。本文将介绍如何使用Netty5进行TCP通信,包括Netty5的核心组件、常用API的介绍和实际应用案例。同时,本文还将介绍如何使用Netty5提高TCP通信的性能和可靠性。一、Netty5的核心组件Netty5

Java API 开发中使用 Netty 进行 TCP 通信 Java API 开发中使用 Netty 进行 TCP 通信 Jun 18, 2023 pm 11:34 PM

JavaAPI开发中使用Netty进行TCP通信在现代软件开发中,网络通信已成为必不可少的一部分。Netty是一种Java框架,用于在高性能网络应用中有效地进行快速开发。它提供了一组易于使用的API,封装了JavaNIO和其他网络库。在JavaAPI开发中,Netty的优越性可以体现在如下几个方面:高性能Netty的应用程

See all articles