【能用】springboot集成netty,解码器处理数据过长的问题

news2025/1/6 9:09:12

netty解码器处理数据过长的问题

处理数据过长的详细流程

当第一次 decode 调用时,如果数据不完整,decode 方法会直接返回,Netty 会保留 ByteBuf 中的数据。后续数据到达时,会再次调用 decode 方法,ByteBuf 会累积新到达的数据。
一旦 ByteBuf 中的数据足够完整解析,就会成功解析出 CustomMessage 并添加到 out 列表中,传递给后续的 YourBusinessLogicHandler 进行处理。
这样,通过检查数据的完整性并利用 Netty 的数据累积特性,就可以实现当数据过长时等待全部数据到达后再进行解码操作。你可以根据具体的协议和需求调整 CustomDataDecoder 中的长度检查逻辑,确保对不同长度的数据都能正确处理。
请注意,在实际应用中,可能需要考虑长时间等待数据不到达的情况,可以添加超时机制或其他错误处理机制,以避免资源的无限占用。例如,可以在 ChannelHandlerContext 上使用 channel().close() 关闭连接,或者使用 ctx.fireExceptionCaught() 抛出异常进行异常处理。

解决思路

明确协议格式

确定 dataLen 在数据中的具体位置。假设 dataLen 在 cmdLen 和 cmdPayload 之后,并且 cmdPayload 的长度是由 cmdLen 确定的。
需要先读取 cmdLen,然后读取 cmdPayload,接着根据 cmdPayload 的长度找到 dataLen 的位置。

逐步解析数据

按照协议规定的顺序和长度,逐步从 ByteBuf 中读取相应的数据。

示例代码(Java 和 Netty):

public class TcpMessageDecoderHandler extends ByteToMessageDecoder {
    private static final Logger logger = LoggerFactory.getLogger(TcpHandleServiceImpl.class);

    @Override
    protected void decode(ChannelHandlerContext channelHandlerContext, ByteBuf in, List<Object> out) throws Exception {
        try {
            // 检查是否有足够的字节读取 cmdLen
            if (in.readableBytes() < 2) {
                // 等待更多数据,因为 cmdLen 占 2 字节
                return;
            }

            // 读取 cmdLen(大端序)
            int cmdLen = in.readUnsignedShort();

            // 检查是否有足够的字节读取 cmdPayload
            if (in.readableBytes() < cmdLen) {
                // 数据不完整,重置读指针,等待更多数据
                in.resetReaderIndex();
                return;
            }

            // 读取 cmdPayload
            byte[] cmdPayload = new byte[cmdLen];
            in.readBytes(cmdPayload);
            String cmd = new String(cmdPayload, StandardCharsets.UTF_8);

            // 检查是否有足够的字节读取 dataLen
            if (in.readableBytes() < 2) {
                // 等待更多数据,因为 dataLen 占 2 字节
                in.resetReaderIndex();
                return;
            }

            // 读取 dataLen(大端序)
            int dataLen = in.readUnsignedShort();

            logger.info("可读字节:" + in.readableBytes());
            // 检查是否有足够的字节读取 dataPayload
            if (in.readableBytes() < dataLen) {
                // 数据不完整,重置读指针,等待更多数据
                in.resetReaderIndex();
                return;
            }

            // 读取 dataPayload
            byte[] dataPayload = new byte[dataLen];
            in.readBytes(dataPayload);

            // 在这里可以将 cmd 和 dataPayload 封装到自定义对象中,例如 CustomMessage
            TcpMessage tcpMessage = new TcpMessage(cmd, dataPayload);
            out.add(tcpMessage);
        } catch (Exception e){
            logger.error("读取字节流出错:" + e.getMessage());
            // 重置读指针
            in.resetReaderIndex();
        }
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
        logger.error("error:" + cause.getMessage());
    }
}
代码解释

首先检查 ByteBuf 中是否有足够的字节读取 cmdLen,如果不足 2 字节,等待更多数据。
读取 cmdLen,它是 2 字节的大端序数据。
检查是否有足够的字节读取 cmdPayload,如果不足,重置读指针等待更多数据。
读取 cmdPayload 并将其转换为字符串(假设是 UTF-8 编码)。
检查是否有足够的字节读取 dataLen,如果不足,重置读指针等待更多数据。
读取 dataLen,它是 2 字节的大端序数据。
检查是否有足够的字节读取 dataPayload,如果不足,重置读指针等待更多数据。
读取 dataPayload 并存储在字节数组中。
创建 CustomMessage 对象,将 cmd 和 dataPayload 作为成员变量,并将其添加到 out 列表中。

问题

1.如果始终没有接收到足够长度的数据,会卡住

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

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

相关文章

python-redis访问指南

Redis&#xff08;Remote Dictionary Server&#xff09;是一种开源的内存数据结构存储&#xff0c;可用作数据库、缓存和消息代理。它功能强大且灵活&#xff0c;可根据需求调整架构和配置&#xff0c;以高性能、简单易用、支持多种数据结构而闻名&#xff0c;广泛应用于各种场…

Flash Attention V3使用

Flash Attention V3 概述 Flash Attention 是一种针对 Transformer 模型中注意力机制的优化实现&#xff0c;旨在提高计算效率和内存利用率。随着大模型的普及&#xff0c;Flash Attention V3 在 H100 GPU 上实现了显著的性能提升&#xff0c;相比于前一版本&#xff0c;V3 通…

UE5失真材质

渐变材质函数&#xff1a;RadialGradientExponential&#xff08;指数径向渐变&#xff09; 函数使用 UV 通道 0 来产生径向渐变&#xff0c;同时允许用户调整半径和中心点偏移。 用于控制渐变所在的位置及其涵盖 0-1 空间的程度。 基于 0-1 的渐变中心位置偏移。 源自中心的径…

Ansys Aqwa 中 Diffraction Analysis 的疲劳结果

了解如何执行疲劳分析&#xff0c;包括由 Ansys Aqwa 计算的海浪行为。 了解疲劳分析 大多数机器故障是由于负载随时间变化&#xff0c;而不是静态负载。这种失效通常发生在应力水平明显低于材料的屈服强度时。因此&#xff0c;当存在动态载荷时&#xff0c;仅依赖静态失效理…

MT8788安卓核心板_MTK8788核心板参数_联发科模块定制开发

MT8788安卓核心板是一款尺寸为52.5mm x 38.5mm x 2.95mm的高集成度电路板&#xff0c;专为各种智能设备应用而设计。该板卡整合了处理器、图形处理单元(GPU)、LPDDR3内存、eMMC存储及电源管理模块&#xff0c;具备出色的性能与低功耗特性。 这款核心板搭载了联发科的MT8788处理…

【UE5 C++课程系列笔记】19——通过GConfig读写.ini文件

步骤 1. 新建一个Actor类&#xff0c;这里命名为“INIActor” 2. 新建一个配置文件“Test.ini” 添加一个自定义配置项 3. 接下来我们在“INIActor”类中获取并修改“CustomInt”的值。这里定义一个方法“GetINIVariable” 方法实现如下&#xff0c;其中第16行代码用于构建配…

Qt天气预报系统设计界面布局第四部分右边

Qt天气预报系统 1、第四部分右边的第一部分1.1添加控件 2、第四部分右边的第二部分2.1添加控件 3、第四部分右边的第三部分3.1添加控件3.2修改控件名字 1、第四部分右边的第一部分 1.1添加控件 拖入一个widget&#xff0c;改名为widget04r作为第四部分的右边 往widget04r再拖…

机器学习基础-机器学习的常用学习方法

半监督学习的概念 少量有标签样本和大量有标签样本进行学习&#xff1b;这种方法旨在利用未标注数据中的结构信息来提高模型性能&#xff0c;尤其是在标注数据获取成本高昂或困难的情况下。 规则学习的概念 基本概念 机器学习里的规则 若......则...... 解释&#xff1a;如果…

搭建nginx文件服务器

1、创建一个nginx配置文件/etc/nginx/nginx.conf user nginx; worker_processes 1;error_log /var/log/nginx/error.log warn; pid /var/run/nginx.pid;events {worker_connections 1024; }http {include mime.types;default_type application/octet-stream;server {li…

MySql---进阶篇(六)---SQL优化

6.1&#xff1a;insert的优化&#xff1a; (1)普通的插入数据 如果我们需要一次性往数据库表中插入多条记录&#xff0c;可以从以下三个方面进行优化。 insert into tb_test values(1,tom); insert into tb_test values(2,cat); insert into tb_test values(3,jerry); 1). 优…

逐光的黑色羽翼:一位黑色超模的成长之路-中小企实战运营和营销工作室博客

逐光的黑色羽翼&#xff1a;一位黑色超模的成长之路-中小企实战运营和营销工作室博客 在遥远的非洲肯尼亚&#xff0c;有一个小女孩名叫艾拉。她生活在一个小小的部落村庄里&#xff0c;每天伴随着朝阳起床&#xff0c;跟着家人放牧&#xff0c;在广袤无垠的草原上奔跑嬉戏&am…

Java项目实战II基于微信小程序的家庭大厨(开发文档+数据库+源码)

目录 一、前言 二、技术介绍 三、系统实现 四、核心代码 五、源码获取 全栈码农以及毕业设计实战开发&#xff0c;CSDN平台Java领域新星创作者&#xff0c;专注于大学生项目实战开发、讲解和毕业答疑辅导。 一、前言 在快节奏的生活中&#xff0c;家庭聚餐成为了连接亲情…

Github拉取项目报错解决

前言 昨天在拉取github上面的项目报错了&#xff0c;有好几个月没用github了&#xff0c;命令如下&#xff1a; git clone gitgithub.com:zhszstudy/git-test.git报错信息&#xff1a; ssh: connect to host github.com port 22: Connection timed out fatal: Could not rea…

学AI编程的Prompt工程,豆包Marscode

学习链接&#xff1a;Datawhale-AI活动https://www.datawhale.cn/activity/116/23/95?rankingPage1 目录 一、如何使用 二、编写游戏 2.1 创意输入与代码生成 2.2 项目初始化与应用 2.3 创意优化与迭代 三、效果展示 一、如何使用 建议在在vscode上安装marscode插件&a…

NLP CH3复习

CH3 3.1 几种损失函数 3.2 激活函数性质 3.3 哪几种激活函数会发生梯度消失 3.4 为什么会梯度消失 3.5 如何解决梯度消失和过拟合 3.6 梯度下降的区别 3.6.1 梯度下降&#xff08;GD&#xff09; 全批量&#xff1a;在每次迭代中使用全部数据来计算损失函数的梯度。计算成本…

计算机网络 (19)扩展的以太网

前言 以太网&#xff08;Ethernet&#xff09;是一种局域网&#xff08;LAN&#xff09;技术&#xff0c;它规定了包括物理层的连线、电子信号和介质访问层协议的内容。以太网技术不断演进&#xff0c;从最初的10Mbps到如今的10Gbps、25Gbps、40Gbps、100Gbps等&#xff0c;已成…

JavaVue-Get请求 数组参数(qs格式化前端数据)

前言 现在管理系统&#xff0c;像若依&#xff0c;表格查询一般会用Get请求&#xff0c;把页面的查询条件传递给后台。其中大部分页面会有日期时间范围查询这时候&#xff0c;为了解决请求参数中的数组文件&#xff0c;前台就会在请求前拦截参数中的日期数组数据&#xff0c;然…

.e01, ..., .e0n的分卷压缩包怎么解压

用BandiZip&#xff0c;这些分卷压缩中还有一个.exe的文件&#xff0c;这个不是可执行文件&#xff0c;是一个解压缩的开头。 安装好bandiZip后&#xff0c;右键这个.exe文件 点击打开就是开始解压了&#xff1a; 最后解压后是这些。然后一个个再次解压.

库伦值自动化功耗测试工具

1. 功能介绍 PlatformPower工具可以自动化测试不同场景的功耗电流&#xff0c;并可导出为excel文件便于测试结果分析查看。测试同时便于后续根据需求拓展其他自动化测试用例。 主要原理&#xff1a;基于文件节点 coulomb_count 实现&#xff0c;计算公式&#xff1a;电流&…

大模型 LangChain 开发框架:Runable 与 LCEL 初探

大模型 LangChain 开发框架&#xff1a;Runable 与 LCEL 初探 一、引言 在大模型开发领域&#xff0c;LangChain 作为一款强大的开发框架&#xff0c;为开发者提供了丰富的工具和功能。其中&#xff0c;Runnable 接口和 LangChain 表达式语言&#xff08;LCEL&#xff09;是构…