TCP协议传输中的粘包和拆包

2024-01-03 01:44
文章标签 协议 tcp 传输 粘包 拆包

本文主要是介绍TCP协议传输中的粘包和拆包,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一 TCP协议的粘包和拆包说明

        经常用tcp协议开发上位机或下位机的同事都会或多或少遇到粘包和拆包的问题,这是在网络通信中最常见的两个问题,这也与数据的发送和接收方式有关。

1 TCP粘包

         指的是发送方发送的多个小数据包被接收方一次性接收的情况。这可能是因为发送方发送数据的速度过快,接收方无法及时处理,从而多个数据包被合并成一个大的数据包一起接收。

2 TCP拆包

        指发送方发送的一个大数据包被接收方拆分成多个小的数据包接收的情况。这可能是由于网络中的路由器、交换机等设备的限制,导致大的数据包在传输过程中被分割成多个小的数据包。接收方需要能够正确地组装这些小数据包以还原原始的大数据包。

二 粘包与拆包造成的原因

        粘包与拆包造成的原因是因为TCP是面向流的协议,数据是一连串的字节流,而不是消息边界明确的数据包

三 解决粘包

1 定长消息:

    在每个消息之前添加消息长度的信息,接收方根据长度信息判断消息的边界。

     java代码示例:

// 发送端
int messageLength = 8; // 消息长度
String message = "68123456";
String formattedMessage = String.format("%-" + messageLength + "s", message);
outputStream.write(formattedMessage.getBytes());// 接收端
byte[] buffer = new byte[messageLength];
int bytesRead = inputStream.read(buffer);
String receivedMessage = new String(buffer, 0, bytesRead);

2 分隔符:  

    使用特定的分隔符(如换行符或其他特殊字符)来标志消息的结束。

    java代码示例:

// 发送端
String message = "68123456\n";
outputStream.write(message.getBytes());// 接收端
BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));
String receivedMessage = reader.readLine();

3 消息头部包含长度信息: 

    在消息头部添加长度字段,表示后续消息的长度。

// 发送端
String message = "68123456";
int messageLength = message.length();
outputStream.writeInt(messageLength);
outputStream.write(message.getBytes());// 接收端
int receivedLength = inputStream.readInt();
byte[] buffer = new byte[receivedLength];
int bytesRead = inputStream.read(buffer);
String receivedMessage = new String(buffer, 0, bytesRead);

4 正则匹配: 

    使用正则表达式处理(不推荐),接入设备不同品牌可能存在上传的数据不是很标准的情况下可以使用完善处理,兼容报文数据格式。

      // 模拟接收到的数据String receivedData = "68123456PP68123456PP68123456";// 定义正则表达式,假设消息之间使用 "PP" 作为分隔符String regex = "PP";Pattern pattern = Pattern.compile(regex);Matcher matcher = pattern.matcher(receivedData);// 使用正则表达式切分消息while (matcher.find()) {String message = receivedData.substring(matcher.start(), matcher.end());System.out.println("Received message: " + message);}

 

这篇关于TCP协议传输中的粘包和拆包的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/564235

相关文章

HTTP 与 SpringBoot 参数提交与接收协议方式

《HTTP与SpringBoot参数提交与接收协议方式》HTTP参数提交方式包括URL查询、表单、JSON/XML、路径变量、头部、Cookie、GraphQL、WebSocket和SSE,依据... 目录HTTP 协议支持多种参数提交方式,主要取决于请求方法(Method)和内容类型(Content-Ty

Java对接MQTT协议的完整实现示例代码

《Java对接MQTT协议的完整实现示例代码》MQTT是一个基于客户端-服务器的消息发布/订阅传输协议,MQTT协议是轻量、简单、开放和易于实现的,这些特点使它适用范围非常广泛,:本文主要介绍Ja... 目录前言前置依赖1. MQTT配置类代码解析1.1 MQTT客户端工厂1.2 MQTT消息订阅适配器1.

Linux中的自定义协议+序列反序列化用法

《Linux中的自定义协议+序列反序列化用法》文章探讨网络程序在应用层的实现,涉及TCP协议的数据传输机制、结构化数据的序列化与反序列化方法,以及通过JSON和自定义协议构建网络计算器的思路,强调分层... 目录一,再次理解协议二,序列化和反序列化三,实现网络计算器3.1 日志文件3.2Socket.hpp

Linux中的HTTPS协议原理分析

《Linux中的HTTPS协议原理分析》文章解释了HTTPS的必要性:HTTP明文传输易被篡改和劫持,HTTPS通过非对称加密协商对称密钥、CA证书认证和混合加密机制,有效防范中间人攻击,保障通信安全... 目录一、什么是加密和解密?二、为什么需要加密?三、常见的加密方式3.1 对称加密3.2非对称加密四、

Linux之UDP和TCP报头管理方式

《Linux之UDP和TCP报头管理方式》文章系统讲解了传输层协议UDP与TCP的核心区别:UDP无连接、不可靠,适合实时传输(如视频),通过端口号标识应用;TCP有连接、可靠,通过确认应答、序号、窗... 目录一、关于端口号1.1 端口号的理解1.2 端口号范围的划分1.3 认识知名端口号1.4 一个进程

Linux中压缩、网络传输与系统监控工具的使用完整指南

《Linux中压缩、网络传输与系统监控工具的使用完整指南》在Linux系统管理中,压缩与传输工具是数据备份和远程协作的桥梁,而系统监控工具则是保障服务器稳定运行的眼睛,下面小编就来和大家详细介绍一下它... 目录引言一、压缩与解压:数据存储与传输的优化核心1. zip/unzip:通用压缩格式的便捷操作2.

如何在Spring Boot项目中集成MQTT协议

《如何在SpringBoot项目中集成MQTT协议》本文介绍在SpringBoot中集成MQTT的步骤,包括安装Broker、添加EclipsePaho依赖、配置连接参数、实现消息发布订阅、测试接口... 目录1. 准备工作2. 引入依赖3. 配置MQTT连接4. 创建MQTT配置类5. 实现消息发布与订阅

使用Python进行GRPC和Dubbo协议的高级测试

《使用Python进行GRPC和Dubbo协议的高级测试》GRPC(GoogleRemoteProcedureCall)是一种高性能、开源的远程过程调用(RPC)框架,Dubbo是一种高性能的分布式服... 目录01 GRPC测试安装gRPC编写.proto文件实现服务02 Dubbo测试1. 安装Dubb

SpringBoot快速搭建TCP服务端和客户端全过程

《SpringBoot快速搭建TCP服务端和客户端全过程》:本文主要介绍SpringBoot快速搭建TCP服务端和客户端全过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录TCPServerTCPClient总结由于工作需要,研究了SpringBoot搭建TCP通信的过程

Nginx中配置HTTP/2协议的详细指南

《Nginx中配置HTTP/2协议的详细指南》HTTP/2是HTTP协议的下一代版本,旨在提高性能、减少延迟并优化现代网络环境中的通信效率,本文将为大家介绍Nginx配置HTTP/2协议想详细步骤,需... 目录一、HTTP/2 协议概述1.HTTP/22. HTTP/2 的核心特性3. HTTP/2 的优