一.包头结构体

  1. //包头
  2. struct PackageHeader
  3. {
  4. //包头大小(sizeof(PackageHeader))
  5. unsigned int uTransPackageHdrSize;
  6. //当前包头的大小(sizeof(PackageHeader)+当前数据包长度)
  7. unsigned int uTransPackageSize;
  8. //数据的总大小
  9. unsigned int uDataSize;
  10. //数据被分成包的个数
  11. unsigned int uDataPackageNum;
  12. //数据包当前的帧号
  13. unsigned int uDataPackageCurrIndex;
  14. //数据包在整个数据中的偏移
  15. unsigned int uDataPackageOffset;
  16. };

每包数据由包头和包体组成,包头用于标记每包数据的特征,包体是图片根据指定大小分出的每段数据,这里指定大小为const int UDP_MAX_SIZE=1200。

为何要分包可参考:TCP、UDP数据包大小的限制

二.分包与组包

1.分包

  1. int dataLength=buffer.data().size();
  2. unsigned char *dataBuffer=(unsigned char *)buffer.data().data();
  3. int packetNum = 0;
  4. int lastPaketSize = 0;
  5. packetNum = dataLength / UDP_MAX_SIZE;
  6. lastPaketSize = dataLength % UDP_MAX_SIZE;
  7. int currentPacketIndex = 0;
  8. if (lastPaketSize != 0)
  9. {
  10. packetNum = packetNum + 1;
  11. }
  12. PackageHeader packageHead;
  13. packageHead.uTransPackageHdrSize=sizeof(packageHead);
  14. packageHead.uDataSize = dataLength;
  15. packageHead.uDataPackageNum = packetNum;
  16. unsigned char frameBuffer[1024*1000];
  17. memset(frameBuffer,0,1024*1000);
  18. while (currentPacketIndex < packetNum)
  19. {
  20. if (currentPacketIndex < (packetNum-1))
  21. {
  22. packageHead.uTransPackageSize = sizeof(PackageHeader)+UDP_MAX_SIZE;
  23. packageHead.uDataPackageCurrIndex = currentPacketIndex+1;
  24. packageHead.uDataPackageOffset = currentPacketIndex*UDP_MAX_SIZE;
  25. memcpy(frameBuffer, &packageHead, sizeof(PackageHeader));
  26. memcpy(frameBuffer+sizeof(PackageHeader), dataBuffer+packageHead.uDataPackageOffset, UDP_MAX_SIZE);
  27. int length=udpsocketSend->writeDatagram(
  28. (const char*)frameBuffer, packageHead.uTransPackageSize,
  29. QHostAddress(ui->lineEditIP->text()), ui->lineEditPort->text().toInt());
  30. if(length!=packageHead.uTransPackageSize)
  31. {
  32. qDebug()<<"Failed to send image";
  33. }
  34. currentPacketIndex++;
  35. }
  36. else
  37. {
  38. packageHead.uTransPackageSize = sizeof(PackageHeader)+(dataLength-currentPacketIndex*UDP_MAX_SIZE);
  39. packageHead.uDataPackageCurrIndex = currentPacketIndex+1;
  40. packageHead.uDataPackageOffset = currentPacketIndex*UDP_MAX_SIZE;
  41. memcpy(frameBuffer, &packageHead, sizeof(PackageHeader));
  42. memcpy(frameBuffer+sizeof(PackageHeader), dataBuffer+packageHead.uDataPackageOffset, dataLength-currentPacketIndex*UDP_MAX_SIZE);
  43. int length=udpsocketSend->writeDatagram(
  44. (const char*)frameBuffer, packageHead.uTransPackageSize,
  45. QHostAddress(ui->lineEditIP->text()), ui->lineEditPort->text().toInt());
  46. if(length!=packageHead.uTransPackageSize)
  47. {
  48. qDebug()<<"Failed to send image";
  49. }
  50. currentPacketIndex++;
  51. }
  52. }

先将图片转换为QBuffer,这样就可以获取图片的数据和长度,然后根据指定大小UDP_MAX_SIZE分包。在包头中设置每包数据的特征,然后将包体加到包头后面发送出去。

2.组包

  1. static int num = 1;
  2. static uint size = 0;
  3. PackageHeader *packageHead = (PackageHeader *)datagram.data();
  4. if (packageHead->uDataPackageCurrIndex == num)
  5. {
  6. num++;
  7. size += packageHead->uTransPackageSize-packageHead->uTransPackageHdrSize;
  8. if (size > 1024*1000)
  9. {
  10. qDebug()<<"image too big";
  11. num = 1;
  12. size = 0;
  13. return;
  14. }
  15. if (packageHead->uDataPackageOffset > 1024*1000)
  16. {
  17. qDebug()<<"image too big";
  18. packageHead->uDataPackageOffset = 0;
  19. num = 1;
  20. size = 0;
  21. return;
  22. }
  23. memcpy(imageData.data+packageHead->uDataPackageOffset, datagram.data()+packageHead->uTransPackageHdrSize,
  24. packageHead->uTransPackageSize-packageHead->uTransPackageHdrSize);
  25. if ((packageHead->uDataPackageNum == packageHead->uDataPackageCurrIndex)
  26. && (size == packageHead->uDataSize))
  27. {
  28. imageData.length = packageHead->uDataSize;
  29. QImage image;
  30. image.loadFromData((uchar *)imageData.data,imageData.length,"JPG");
  31. QPixmap pixmap=QPixmap::fromImage(image);
  32. ui->labelImage_2->setPixmap(pixmap);
  33. recvImageNum++;
  34. ui->lineEditRevFrame->setText(QString::number(recvImageNum));
  35. ui->lineEditRevSize->setText(QString::number(imageData.length));
  36. memset(&imageData,0,sizeof(UdpUnpackData));
  37. num = 1;
  38. size = 0;
  39. }
  40. }
  41. else
  42. {
  43. num = 1;
  44. size = 0;
  45. memset(&imageData,0,sizeof(UdpUnpackData));
  46. }

组包是分包的逆过程,根据包头中“数据的总大小”和“数据被分成包的个数”两个字段可以判断组包是否完整,如果完整就在接收端显示出来。
三.示例

界面包括发送端和接收端,接收端的IP地址是自动获取的本机IP地址,上图将发送端的IP地址设置为与接收端IP地址相同,可实现自发自收。

每秒帧数可实时设置每秒发送图片的张数,发送帧数表示一共发送了多少张,接收帧数表示一共接收了多少张,图片大小表示每张图片多少Byte。

可将示例运行于两台计算机,实现双向收发。

源码链接:见http://blog.csdn.net/caoshangpa/article/details/52681572的评论

http://blog.csdn.net/caoshangpa/article/details/52681572

Qt通过UDP传图片,实现自定义分包和组包的更多相关文章

  1. 异步SOCKET分包和组包的一种通用算法

    unit uPackage;// 应用协议// cxg 2016-9-23// 包=包头+包体 interface uses SysUtils, Classes, PeachCtrl.Net.Iocp ...

  2. Socket之UDP分包组包

    一般传输大的文件和信息的时候需要涉及到分包和组包,方法有很多,下面一种是借鉴了别人的思路,供大家参考哈 分包 1.取出需要传输的文件和字符的长度和大小放入缓存区里面: 2.设定固定传输的长度,用需要传 ...

  3. 使用 Qt 获取 UDP 数据并显示成图片

    一个项目,要接收 UDP 数据包,解析并获取其中的数据,主要根据解析出来的行号和序号将数据拼接起来,然后将拼接起来的数据(最重要的数据是 R.G.B 三个通道的像素值)显示在窗口中.考虑到每秒钟要接收 ...

  4. 使用 Qt 获取 UDP 数据并显示成图片(2)

    本文首发于 BriFuture 的 个人博客 在我的前一篇文章 使用 Qt 获取 UDP 数据并显示成图片 中,我讲了如何用 Python 模拟发送数据,如何在 Qt 中高效的接收 UDP 数据包并将 ...

  5. 7.Vue-Quill-Editor图片插入自定义

    Vue-Quill-Editor图片插入自定义 前言: 因为在项目中前端采用了Vue来实现,正好用到了富文本编辑器这一块,于是,经过技术上的选择,决定使用Vue-Quill-Editor. 使用的过程 ...

  6. Qt自适应大小显示图片,添加菜单

    由于后面的图像处理需要UI,OpenCV自带也不怎么会,MFC实在懒得学的.听同学说Qt不错,就用Qt做UI了. 本文主要介绍三个内容:在Qt Creator中使用OpenCV2.Qt中自适应显示图片 ...

  7. iOS开发——UI进阶篇(十八)核心动画小例子,转盘(裁剪图片、自定义按钮、旋转)图片折叠、音量震动条、倒影、粒子效果

    一.转盘(裁剪图片.自定义按钮.旋转) 1.裁剪图片 将一张大图片裁剪为多张 // CGImageCreateWithImageInRect:用来裁剪图片 // image:需要裁剪的图片 // re ...

  8. 用qt代码怎样编写图片保存格式[qt4.6]

    用qt代码怎样编写图片保存格式 qt提供了多个保存图片的接口,比较常用的接口如下 bool QPixmap::save ( const QString & fileName, const ch ...

  9. iOS-AFNetworking-上传图片

    一.AFNetworking上传图片 /** 需要Demo,可留言 */ 1.利用AFnetworking上传图片. 2.注意将UIImage类型转化为NSData类型,传给后台. 二.Demo // ...

随机推荐

  1. android NDK开发环境搭建

    android NDK开发环境搭建 2012-05-14 00:13:58 分类: 嵌入式 基于 Android NDK 的学习之旅-----环境搭建 工欲善其事必先利其器 , 下面介绍下 Eclip ...

  2. VS中引用第三方库的方法(配置sqlite数据库)

    我们在编写程序时,,不可避免的会使用第三方的库文件,很少使用源文件(.cpp),大部分是使用对类进行声明的头文件和封装了类的链接库(静态lib或动态dll),比如我们写程序用的iostream这个库, ...

  3. 硬件相关-EMI & EMS & EMC

    EMI——Electro Magnetic Interference 电磁干扰 定义:是指电磁波与电子元件作用后而产生的干扰现象. 分类:有传导干扰和辐射干扰两种. 传导干扰: 是指通过导电介质把一个 ...

  4. IO和NIO的区别

    http://my.oschina.net/u/1010990/blog/192558 传统的socket IO中,需要为每个连接创建一个线程,当并发的连接数量非常巨大时,线程所占用的栈内存和CPU线 ...

  5. 向Array中添加归并排序

    归并排序思路 1) 归并 从两个有序表R[low...mid]和R[mid+1...high],每次从左边依次取出一个数进行比较,将较小者放入tmp数组中,最后将两段中剩下的部分直接复制到tmp中. ...

  6. block extends include三者的差别跟用法

    block extends include三者的差别跟用法 一.定义基础模板,在html内容中定义多个block块,block由子模板引用同名block块,来决定是否替换这些部分{% block ti ...

  7. Dijkstra--POJ 2502 Subway(求出所有路径再求最短路径)

    题意: 你从家往学校赶,可以用步行和乘坐地铁这两种方式,步行速度为10km/h,乘坐地铁的速度为40KM/h.输入数据的第一行数据会给你起点和终点的x和y的坐标.然后会给你数目不超过200的双向地铁线 ...

  8. hdu 1392 Surround the Trees

    题目地址:http://acm.hdu.edu.cn/showproblem.php?pid=1392 题意:给出一些点的坐标,求最小的凸多边形把所有点包围时此多边形的周长. 解法:凸包ConvexH ...

  9. JavaScript之表单验证讲解

    JavaScript 可用来在数据被送往服务器前对 HTML 表单中的这些输入数据进行验证. JavaScript 表单验证 JavaScript 可用来在数据被送往服务器前对 HTML 表单中的这些 ...

  10. codeforces #240 div 2

    A:语文题,估计大家都会, B题:假如答案是ans,求最大的ans,是w*a/b==(w-ans)*a/b; 明显的二分,可是我的二分写的没水准,还有是直接做: #include<string. ...