博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
golang中解决tcp传输中的粘包问题
阅读量:6235 次
发布时间:2019-06-22

本文共 3816 字,大约阅读时间需要 12 分钟。

golang中解决tcp传输中的粘包问题

Author: 岳东卫

Email: usher.yue@gmail.com

什么是粘包?

最近在写 (一个http/https反向代理服务)的时候遇到了粘包问题,

如果有做过网络编程的小伙伴应该都知道粘包问题,举个例子: 比如客户端在和服

务器进行通信采用的是json格式的数据包。那么此时Client和Server的数据交互流程应该如下:

Client Send Json Data->经过网络->Server Reveive Data->Server Decode Json ->Done (一次交互只有一个Json数据包)

上述流程我们假设从客户端发送到服务器接收这个一次的性动作中间交互的数

据是一个完成的json数据包,因此我们的程序可以正常工作。

但是实际情况并不是我们想的这样,由于TCP协议的特点、以及网络环境的复杂

多变、以及服务器对客户端的数据接收处理不及时等等原因,会导致网络传输

过程中出现粘包。 也就是说在服务器进行一次数据读取的时候,我们假想这

个数据包是一个完整的json数据包,但是实际上他确实 ,2个Json 数据包、3

个json数据包、2.5个json数据包,这就是我们所说的粘包。

如果你还不能理解那么看下图。

这里写图片描述

我们如何解决粘包问题?

我们在开发过程中通常会在server端接收数据的时候定义一个固定长度的buffer来存储从客户端连接发来的数据包 ,然后对这个数据包进行反序列化,所以要解决这个问题我们就要从收发数据的时候做一些手脚, 思路如下:

Client Send Json Data->调用封装方法将数据封装成固定格式的Packet->经过网络->Server Reveive Data->调用解封装方法取出粘包packet中所有json数据包,并将剩余截断数据和下一次到来的数据包进行拼接->Server Decode Json ->Done (一次交互只有一个Json数据包)

我在golang中实现了一个Packet封装代码如下,可直接使用:

package packetimport (    "bytes"    "encoding/binary")const (    DEFAULE_HEADER           = "[**********]"    DEFAULT_HEADER_LENGTH    = 12    DEFAULT_SAVE_DATA_LENGTH = 4)type Packet struct {    Header         string    HeaderLengh    int32    SaveDataLength int32    Data           []byte}//set delimiter headerfunc (self *Packet) SetHeader(header string) *Packet {    self.Header = header    self.HeaderLengh = int32(len([]byte(header)))    return self}//create default packagefunc NewDefaultPacket(data []byte) *Packet {    return &Packet{DEFAULE_HEADER, DEFAULT_HEADER_LENGTH, DEFAULT_SAVE_DATA_LENGTH, data}}//convert to net packagefunc (self *Packet) Packet() []byte {    return append(append([]byte(self.Header), self.IntToBytes(int32(len(self.Data)))...), self.Data...)}//return value is sticky datafunc (self *Packet) UnPacket(readerChannel chan []byte) []byte {    dataLen := int32(len(self.Data))    var i int32    for i = 0; i < dataLen; i++ {        //Termiate for loop when the remaining data is insufficient .        if dataLen < i+self.HeaderLengh+self.SaveDataLength {            break        }        //find Header        if string(self.Data[i:i+self.HeaderLengh]) == self.Header {            saveDataLenBeginIndex := i + self.HeaderLengh            actualDataLen := self.BytesToInt(self.Data[saveDataLenBeginIndex : saveDataLenBeginIndex+self.SaveDataLength])            //The remaining data is less than one package            if dataLen < i+self.HeaderLengh+self.SaveDataLength+actualDataLen {                break            }            //Get a packet            packageData := self.Data[saveDataLenBeginIndex+self.SaveDataLength : saveDataLenBeginIndex+self.SaveDataLength+actualDataLen]            //send pacakge data to reader channel            readerChannel <- packageData            //get next package index            i += self.HeaderLengh + self.SaveDataLength + actualDataLen - 1        }    }    //Reach the end    if i >= dataLen {        return []byte{}    }    //Returns the remaining data    return self.Data[i:]}func (self *Packet) IntToBytes(i int32) []byte {    byteBuffer := bytes.NewBuffer([]byte{})    binary.Write(byteBuffer, binary.BigEndian, i)    return byteBuffer.Bytes()}func (self *Packet) BytesToInt(data []byte) int32 {    var val int32    byteBuffer := bytes.NewBuffer(data)    binary.Read(byteBuffer, binary.BigEndian, &val)    return val}

Client实现伪代码代码如下:

dataPackage := NewDefaultPacket([]byte(jsonString)).Packet()  Client.Write(dataPackage)

Server实现伪代码代码如下:

//Declare a pipe for receiving unpacked data    readerChannel := make(chan []byte, 1024)    //Store truncated data    remainBuffer := make([]byte, 0)    //read unpackage data from buffered channel    go func(reader chan []byte) {        for {            packageData := <-reader            //....balabala....        }    }(readerChannel)  remainBuffer =   NewDefaultPacket(append(remainBuffer,recvData)).UnPacket(readerChannel)
你可能感兴趣的文章
centos7的使用
查看>>
【持续更新】IDEA常用快捷键
查看>>
CentOS 编译安装新版git
查看>>
12.6 Nginx安装 12.7 默认虚拟主机 12.8 Nginx用户认证12.9 Nginx域名重定向
查看>>
tomcat 启动和关闭源码查看
查看>>
JavaScript设计模式之观察者模式
查看>>
osx中让idea使用官方版的git
查看>>
js 数组 map方法
查看>>
Linux 工程师技术
查看>>
Apk代码混淆
查看>>
线程池监控
查看>>
php源码编译常见错误解决方案
查看>>
ios 开发中UISegmentedControl 用法
查看>>
分布式网站架构后续:zookeeper技术浅析
查看>>
Redis学习(二)—— 数据类型(1)
查看>>
Darwin Streaming Server 核心代码分析
查看>>
Linux系统安装
查看>>
WordPress 后台禁用Google Open Sans字体,加速网站
查看>>
如何获取好链接??(下)
查看>>
Javascript与Ajax
查看>>