【转】Socket接收字节缓冲区
原创本拉灯
2014年04月16日 10:06:55
我们接收Socket字节流数据一般都会定义一个数据包协议( 协议号,长度,内容),由于Socket接收数据是连续的,对方发两个包过来,Socket的 Recive事件有可能只触发一次或触发三次,也就是大家听到的粘包,为解决这个粘包,所以我们必要建一个字节缓冲区,将所有的接收到的字节流全放到这个缓冲区内 由这个缓冲区来分隔每个数据包的内容。
这份代码也是为论坛某个人解决串口接收数据包时而写的。不多说了上代码:
- /// <summary>
- /// 字节缓冲器
- /// </summary>
- public class ByteQueue
- {
- private List<byte> m_buffer = new List<byte>();
- public bool Find()
- {
- if (m_buffer.Count == 0)
- return false;
- int HeadIndex = m_buffer.FindIndex(o => o == 0xAA);
- if (HeadIndex == -1)
- {
- m_buffer.Clear();
- return false; //没找到AA
- }
- else if (HeadIndex != 0) //不为开头移掉之前的字节
- {
- if (HeadIndex > 1)
- m_buffer.RemoveRange(0, HeadIndex);
- }
- int length= GetLength();
- if (m_buffer.Count <length)
- {
- return false;
- }
- int TailIndex = m_buffer.FindIndex(o => o == 0x55); //查找55的位置
- if (TailIndex == -1)
- {
- //这一步为防止连发一个AA开头的包后,没发55,而又发了一个AA
- int head = m_buffer.FindLastIndex(o => o == 0xAA);
- if (head > -1)
- {
- m_buffer.RemoveRange(0, head);
- }
- return false;
- }
- else if (TailIndex + 1 != length) //计算包尾是否与包长度相等
- {
- m_buffer.RemoveRange(0, TailIndex);
- return false;
- }
- return true;
- }
- /// <summary>
- /// 命令类型
- /// </summary>
- /// <returns></returns>
- public byte Cmd()
- {
- if (m_buffer.Count >= 2)
- {
- return m_buffer[1];
- }
- return 0;
- }
- /// <summary>
- /// 序号
- /// </summary>
- /// <returns></returns>
- public byte Number()
- {
- if (m_buffer.Count >= 3)
- {
- return m_buffer[2];
- }
- return 0;
- }
- /// <summary>
- /// 包长度
- /// </summary>
- /// <returns></returns>
- public int GetLength()
- {
- int len = 5;//AA 命令类型 序号 校验和 55
- if (m_buffer.Count >= 3)
- {
- switch (m_buffer[2]) //第三字节为序号
- {
- case 0x00: //序号
- return len + 16;
- case 0x01: //序号
- return len + 10;
- case 0x02: //序号
- return len + 12;
- }
- }
- return 0;
- }
- /// <summary>
- /// 提取数据
- /// </summary>
- public void Dequeue(byte[] buffer, int offset,int size)
- {
- m_buffer.CopyTo(0,buffer,offset,size);
- m_buffer.RemoveRange(0, size);
- }
- /// <summary>
- /// 队列数据
- /// </summary>
- /// <param name="buffer"></param>
- public void Enqueue(byte[] buffer)
- {
- m_buffer.AddRange(buffer);
- }
- }
调用列子:
- private ByteQueue queue = new ByteQueue();
- private void serialPort1_DataReceived(object sender, SerialDataReceivedEventArgs e)
- {
- int len = serialPort1.BytesToRead;
- if (len > 0)
- {
- byte[] temp = new byte[len];
- serialPort1.Read(temp, 0, len);
- queue.Enqueue(temp);
- while (queue.Find()) //while可处理同时接收到多个AA ... 55 ,AA...55的包
- {
- int length = queue.GetLength();
- byte[] readBuffer = new byte[len];
- queue.Dequeue(readBuffer, 0, length);
- OnReceiveData(readBuffer); //<这里自己写一个委托吧就OK了
- }
- }
- }
上面的字节接收容器是用List来处理为方便进出字节后移除整个数据包的字节数据,当然更高效的应用byte[] 数组作成环形缓冲会好很多相对应的写法也会难一些,
【转】Socket接收字节缓冲区的更多相关文章
- 网络编程基础【day09】:socket接收大数据(五)
本节内容 1.概述 2.socket接收大数据 3.中文字符的坑 一.概述 上篇博客写到了,就是说当服务器发送至客户端的数据,大于客户端设置的数据,则就会把数据服务端发过来的数据剩余数据存在IO缓冲区 ...
- 【python】-- Socket接收大数据
Socket接收大数据 上一篇博客中的简单ssh实例,就是说当服务器发送至客户端的数据,大于客户端设置的数据,则就会把数据服务端发过来的数据剩余数据存在IO缓冲区中,这样就会造成我们想要获取数据的完整 ...
- Python网络编程——修改套接字发送和接收的缓冲区大小
很多情况下,默认的套接字缓冲区大小可能不够用.此时,可以将默认的套接字缓冲区大小改成一个更合适的值. 1. 代码 # ! /usr/bin/env python # -*- coding: utf-8 ...
- android开发 socket接收图片并保存
逻辑:接收到socket之后需要将socket发送的图片数据保存下来并通知handler更新界面 关键代码: public void readImage(Socket socket) { try { ...
- Java基础知识强化之IO流笔记28:BufferedOutputStream / BufferedInputStream(字节缓冲区流) 之BufferedOutputStream写出数据
1. BufferedOutputStream / BufferedInputStream(字节缓冲区流)的概述 通过定义数组的方式确实比以前一次读取一个字节的方式快很多,所以,看来有一个缓冲区还是非 ...
- Java中的字节流,字符流,字节缓冲区,字符缓冲区复制文件
一:创建方式 1.建立输入(读)对象,并绑定数据源 2.建立输出(写)对象,并绑定目的地 3.将读到的内容遍历出来,然后在通过字符或者字节写入 4.资源访问过后关闭,先创建的后关闭,后创建的先关闭 ...
- Java基础-虚拟内存之映射字节缓冲区(MappedByteBuffer)
Java基础-虚拟内存之映射字节缓冲区(MappedByteBuffer) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.映射字节缓冲区 1>.什么是虚拟内存 答:虚拟内 ...
- 设置socket接收和发送超时的一种方式
Linux环境设置Socket接收和发送超时: 须如下定义:struct timeval timeout = {3,0}; //设置发送超时setsockopt(socket,SOL_SOCKET, ...
- C# Socket发送接收字节数组和十六16进制之间转换函数
近期在使用远程网络模块的时候, 需要用的Socket发送数据,远程模块指令为16进制. 官方提供的DEMO比较繁琐.不方便新手使用. 下面的转换函数可大大方便新手使用. // 16进制字符串转字节数组 ...
随机推荐
- python的copy模块理解
首先直接上结论: —–我们寻常意义的复制就是深复制,即将被复制对象完全再复制一遍作为独立的新个体单独存在.所以改变原有被复制对象不会对已经复制出来的新对象产生影响. —–而浅复制并不会产生一个独立的对 ...
- Unknown system variable 'query_cache_size'] with root cause
Unknown system variable 'query_cache_size'] with root cause 出现这个错误是因为mysql连接数据库的版本不对, mysql-connecto ...
- 模块--random
random模块 1 random.random() print(random.random()) 0-1 之间随机小数 不包含1 2 random.uniform(a,b) ...
- dsp实验一 常见问题教程
1.选择合适阅读材料:卖油翁.将进酒,等. 2.录音设备三选一:某种电脑声卡+线上convert/楼月mp3录音软件+格式工厂(我选的)/Matlab (注:这是目前已知的方法,我只用了第二个) 3. ...
- day 25-1 接口类、抽象类、多态
# 接口类:python 原生不支持# 抽象类:python 原生支持的 接口类 首先我们来看一个支付接口的简单例子 from abc import abstractmethod,ABCMeta #我 ...
- day 19 - 1 模块
collections 模块 在内置数据类型(dict.list.set.tuple)的基础上,collections 模块还提供了几个额外的数据类型:Counter.deque.defaultdic ...
- day 18 - 2 正则与 re 模块练习
1.爬虫的例子 #爬虫的例子(方法一) import re import urllib,request import urlopen def getPage(url): response = urlo ...
- MongoDB代码——Python篇
需要安装的库:pymongo 一.添加文档 from pymongo import MongoClient # 连接服务器 conn = MongoClient("localhost&quo ...
- centos7.6设置sftp服务
sftp是Secure File Transfer Protocol的缩写,安全文件传送协议. CentOS自带 SSH 服务,直接配置即可 1. 查看ssh版本 sftp是基于ssh协议的,首先查看 ...
- 题解 P5301 【[GXOI/GZOI2019]宝牌一大堆】
这道题除了非常恶心以外也没有什么非常让人恶心的地方 当然一定要说有的话还是有的,就是这题和咱 ZJOI 的 mahjong 真的是好像的说~ 于是就想说这道题出题人应该被 锕 掉 noteskey 整 ...