Netty:java高性能网络编程的基石(下)

一、Netty是什么?为什么需要它?

Netty是一个异步事件驱动的网络应用框架,基于Java NIO技术封装,用于快速开发高性能、高可靠性的网络服务器和客户端程序。作为当前最流行的NIO框架之一,支撑着Dubbo、RocketMQ、Elasticsearch等众多分布式系统的底层通信。

为什么需要Netty?

传统BIO的困境: 阻塞式IO模型(BIO)中每个连接都需要独立线程,海量连接时线程资源耗尽。
NIO的复杂性: 虽然Java原生NIO解决了BIO的阻塞问题,但其Selector、Buffer、Channel等API复杂难用,开发维护成本高。

  • Netty的优势:
    优雅的API设计: 屏蔽NIO底层细节,提供易用的编程模型。
    高性能架构: 基于主从Reactor线程模型,支持百万级并发连接。
    丰富的协议支持: 内置HTTP、WebSocket等协议编解码器,开箱即用。
    健壮性保障: 完善的异常处理机制,避免网络波动导致服务崩溃。
// 传统BIO线程模型
ServerSocket serverSocket = new ServerSocket(8080);
while(true) {Socket socket = serverSocket.accept(); // 阻塞等待连接new Thread(() -> handleRequest(socket)).start(); // 每个连接一个线程
}

二、Netty如何解决粘包/拆包?

粘包/拆包问题根源: TCP是流式协议,像水管里的水,无法区分消息边界。发送方多次写入的数据可能被接收方一次读取(粘包),或一个完整数据包被分多次读取(拆包)。

Netty的解决方案:

1. 固定长度解码器 (FixedLengthFrameDecoder)

每个数据包固定长度,不足则补位。

// 服务端添加解码器
ch.pipeline().addLast(new FixedLengthFrameDecoder(10)); // 每个数据包10字节
ch.pipeline().addLast(new StringDecoder()); 
ch.pipeline().addLast(new SimpleChannelInboundHandler<String>() {@Overrideprotected void channelRead0(ChannelHandlerContext ctx, String msg) {System.out.println("收到消息: " + msg);}
});
2. 分隔符解码器 (DelimiterBasedFrameDecoder)

使用特殊符号(如换行符)作为消息结束标志。

// 使用"$_"作为分隔符
ByteBuf delimiter = Unpooled.copiedBuffer("$_".getBytes());
ch.pipeline().addLast(new DelimiterBasedFrameDecoder(1024, delimiter));
ch.pipeline().addLast(new StringDecoder());
3. 长度字段解码器 (LengthFieldBasedFrameDecoder)

协议头中定义长度字段,指明数据包长度。

// 协议格式:长度字段(4字节) + 数据内容
ch.pipeline().addLast(new LengthFieldBasedFrameDecoder(1024, 0, 4, 0, 4)); // 最大长度1024,长度字段偏移0,长度4字节
ch.pipeline().addLast(new StringDecoder(CharsetUtil.UTF_8));

三、私有协议设计与实现

实现步骤:

1. 定义协议格式

以简单的协议为例:

字段魔数(4字节)版本(1字节)数据长度(4字节)数据内容
示例0xCAFEBABE1lendata
2. 自定义编码器

继承MessageToByteEncoder,将消息对象转为字节流。

public class CustomEncoder extends MessageToByteEncoder<CustomMessage> {@Overrideprotected void encode(ChannelHandlerContext ctx, CustomMessage msg, ByteBuf out) {out.writeInt(0xCAFEBABE); // 魔数out.writeByte(1); // 版本out.writeInt(msg.getData().length); // 数据长度out.writeBytes(msg.getData()); // 数据内容}
}
3. 自定义解码器

继承ReplayingDecoder,解析字节流为消息对象。

public class CustomDecoder extends ReplayingDecoder<Void> {@Overrideprotected void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out) {int magic = in.readInt(); // 读取魔数if(magic != 0xCAFEBABE) {throw new IllegalStateException("协议错误");}byte version = in.readByte(); // 版本校验int length = in.readInt(); // 数据长度byte[] data = new byte[length];in.readBytes(data); // 读取数据out.add(new CustomMessage(version, data));}
}

处理半包问题:ReplayingDecoder内部通过检查可读字节数自动处理数据不完整的情况,当可读字节不足时等待下次数据到达。

四、总结

Netty通过精妙的架构设计和丰富的组件库,极大简化了网络编程的复杂性。掌握粘包/拆包解决方案和自定义协议开发能力,是构建高性能通信系统的关键。建议读者结合官方文档和实际项目案例,深入理解Netty的线程模型、内存管理等高级特性。

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

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

相关文章

leetcode-47.全排列II

如何在有重复值的时候节省时间是优化重点。 基础写法肯定是按无重复值时的全排列写&#xff0c;在其中要加上防止走重复路径的分支。 能防止的也只有同层&#xff0c;如果同层走一个值&#xff0c;但是该值重复&#xff0c;且走过了&#xff0c;则放弃走该分支。所以设layer_u…

函数(函数的概念、库函数、自定义函数、形参和实参、return语句、数组做函数参数、嵌套调用和链式访问、函数的声明和定义、static和extern)

一、函数的概念 •C语⾔中的函数&#xff1a;⼀个完成某项特定的任务的⼀⼩段代码 •函数又被翻译为子函数&#xff08;更准确&#xff09; •在C语⾔中我们⼀般会⻅到两类函数&#xff1a;库函数 ⾃定义函数 二、库函数 1 .标准库和头文件 •C语⾔的国际标准ANSIC规定了⼀…

孜然SEO静态页面生成系统V1.0

孜然SEO静态页面生成系统&#xff0c;1秒生成上万个不同的静态单页系统&#xff0c;支持URL裂变采集&#xff0c;采集的内容不会重复&#xff0c;因为程序系统自带AI重写算法&#xff0c;AI扩写算法&#xff0c;可视化的蜘蛛池系统让您更清楚的获取到信息&#xff01; 可插入二…

Secs/Gem第一讲 · 总结精华版(基于secs4net项目的ChatGpt介绍)

好的&#xff01;这就是《第一讲 总结精华版》——为背诵准备的口述速成稿&#xff0c;适合面试前复习答题用。我们会分为两个部分&#xff1a; 第一部分&#xff1a;一整段口述稿&#xff0c;可以当成面试时开口自我介绍用&#xff1b;第二部分&#xff1a;要点清单关键词串…

预处理指令中#if 和 #endif的用法

在 C 语言中&#xff0c;#if 和 #endif 是预处理指令&#xff0c;用于条件编译。它们的核心作用是&#xff1a;根据预处理器能够识别的条件&#xff08;通常是宏定义或常量表达式&#xff09;&#xff0c;决定某段代码是否参与编译。 — 基本功能 #if 用于开启一个条件编译块…

【数据库】掌握MySQL事务与锁机制-数据一致性的关键

在数据库的世界里&#xff0c;数据就是一切。而确保数据的准确性和一致性&#xff0c;则是数据库系统的核心任务之一。想象一下&#xff0c;如果没有合适的机制&#xff0c;当多个用户同时试图修改同一条数据时&#xff0c;会发生什么&#xff1f; chaos&#xff08;混乱&#…

linux 基础网络配置文件

使用“ifconfig”命令查看网络接口地址 直接执行“iconfg”命令后可以看到ens33、10、virbr0这3个网络接口的信息&#xff0c;具体命令如下 ifconfig ##查看网络接口地址 ens33:第一块以太网卡的名称 lo:“回环”网络接口 virbr0:虚拟网桥的连接接口 查看指…

OpenCV特征提取与深度学习CNN特征提取差异

一、特征生成方式 ‌OpenCV传统方法‌ ‌手工设计特征‌&#xff1a;依赖人工设计的算法&#xff08;如SIFT、FAST、BRIEF&#xff09;提取图像中的角点、边缘等低层次特征&#xff0c;需手动调整参数以适应不同场景‌。‌数学驱动‌&#xff1a;基于梯度变化、几何变换等数学规…

五种方案实现双链路可靠数据传输

本文介绍五种双链路数据传输方案,目标是利用设备的多个传输通道,(如双有线网口,网口+wifi, 网口+5G等场景 , 网口+ 自组网, 自组网 + 5G等),将数据复制后分流、分路同时传输,以期提高数据传输可靠性,满足高可靠性传输的应用场景需求。部分方案给出了实际验证结果 。 …

【备赛】遇到的小问题-1

问题描述-1 想实现的功能是&#xff0c;通过ADC实时测量某引脚的电压及其占空比。 可以通过旋转电位器&#xff0c;更改其电压。 首先我定义了这几个变量 uint32_t adc_value;//HAL库函数里面得出的采样值(实时更新) uint32_t percentage6;//占空比&#xff0c;随着adc_val…

最大公约数

4.最大公约数 - 蓝桥云课 最大公约数 题目描述 给定两个正整数 A,B&#xff0c;求它们的最大公约数。 输入描述 第1行为一个整数 T&#xff0c;表示测试数据数量。 接下来的 T 行每行包含两个正整数 A,B。 1≤T≤105&#xff0c;1≤A,B≤109。 输出描述 输出共 T 行&…

TMHMM2.0-蛋白跨膜螺旋预测工具-centos-安装+配置+排错

参考&#xff1a; A. Krogh, B. Larsson, G. von Heijne, and E. L. L. Sonnhammer. Predicting transmembrane protein topology with a hidden Markov model: Application to complete genomes. Journal of Molecular Biology, 305(3):567-580, January 2001. centos&#x…

docker run 命令常用参数

docker run 命令 用于从镜像创建并启动一个新的容器。 基本语法&#xff1a; docker run [OPTIONS] IMAGE [COMMAND] [ARG...]常用选项分类说明 容器配置 --name 为容器指定名称&#xff08;默认随机生成&#xff09;。 示例&#xff1a; docker run --name my_container …

Zbrush插件安装

安装目录在: ...\Zbrush2022\ZStartup\ZPlugs64

pandas中excel自定义单元格颜色

writerpd.ExcelWriter(filepathf05教师固定学生占比1月{today}.xlsx,engineopenpyxl) df.to_excel(writer,sheet_name明细) piv1.to_excel(writer,sheet_name1月分布) wswriter.book.create_sheet(口径) ws.cell(1,1).value综合占比&#xff1a; ws.cell(1,2).value固定学生占比…

整体二分算法讲解及例题

算法思想 整体二分&#xff0c;带有二分二字那么就一定和二分脱不了干系。 整体二分算法常用来解决询问区间的第 k k k小值的问题&#xff0c;思路如下&#xff1a; 我们二分的对象是这道题目给定的值域&#xff0c;及最小值与最大值之间的区间&#xff0c;在题目给定的数组中…

python+flask实现360全景图和stl等多种格式模型浏览

1. 安装依赖 pip install flask 2. 创建Flask应用 创建一个基本的Flask应用&#xff0c;并设置路由来处理不同的文件类型。 from flask import Flask, render_template, send_from_directory app Flask(__name__) # 设置静态文件路径 app.static_folder static app.r…

XML转义符详解:如何在XML中正确处理特殊字符

在XML中&#xff0c;某些字符具有特殊含义&#xff0c;如果直接在文本中使用这些字符&#xff0c;可能会导致XML解析错误。为了避免这些问题&#xff0c;我们需要使用XML转义符或CDATA段来处理这些特殊字符。本文将详细介绍XML转义符的使用方法、注意事项&#xff0c;并结合实际…

IPC 进程间通信(一):管道(匿名管道进程池)

1. 初识进程间通信 1.1进程间通信的目的&#xff1a; 1、数据传输&#xff1a;一个进程需要将它的数据发送给另一个进程 2、资源共享&#xff1a;多个进程之间共享同样的资源 3、通知事件&#xff1a;一个进程需要向另一个或一组进程发送消息&#xff0c;通知它&#xff08;它…

Linux-数据结构-单链表练习-双链表

一.单链表练习和一些功能实现 【1】单链表实现字典查询 &#xff08;1&#xff09;定义存放数据的结构体&#xff0c;在每次向里面存放数据时候需要清空 &#xff08;2&#xff09;对字典进行切割 空格切割字母&#xff0c;再从剩余里切割到解释&#xff0c;这里windows的txt文…