Qt通过UDP传图片,实现自定义分包和组包
一.包头结构体
- //包头
- struct PackageHeader
- {
- //包头大小(sizeof(PackageHeader))
- unsigned int uTransPackageHdrSize;
- //当前包头的大小(sizeof(PackageHeader)+当前数据包长度)
- unsigned int uTransPackageSize;
- //数据的总大小
- unsigned int uDataSize;
- //数据被分成包的个数
- unsigned int uDataPackageNum;
- //数据包当前的帧号
- unsigned int uDataPackageCurrIndex;
- //数据包在整个数据中的偏移
- unsigned int uDataPackageOffset;
- };
每包数据由包头和包体组成,包头用于标记每包数据的特征,包体是图片根据指定大小分出的每段数据,这里指定大小为const int UDP_MAX_SIZE=1200。
为何要分包可参考:TCP、UDP数据包大小的限制
二.分包与组包
1.分包
- int dataLength=buffer.data().size();
- unsigned char *dataBuffer=(unsigned char *)buffer.data().data();
- int packetNum = 0;
- int lastPaketSize = 0;
- packetNum = dataLength / UDP_MAX_SIZE;
- lastPaketSize = dataLength % UDP_MAX_SIZE;
- int currentPacketIndex = 0;
- if (lastPaketSize != 0)
- {
- packetNum = packetNum + 1;
- }
- PackageHeader packageHead;
- packageHead.uTransPackageHdrSize=sizeof(packageHead);
- packageHead.uDataSize = dataLength;
- packageHead.uDataPackageNum = packetNum;
- unsigned char frameBuffer[1024*1000];
- memset(frameBuffer,0,1024*1000);
- while (currentPacketIndex < packetNum)
- {
- if (currentPacketIndex < (packetNum-1))
- {
- packageHead.uTransPackageSize = sizeof(PackageHeader)+UDP_MAX_SIZE;
- packageHead.uDataPackageCurrIndex = currentPacketIndex+1;
- packageHead.uDataPackageOffset = currentPacketIndex*UDP_MAX_SIZE;
- memcpy(frameBuffer, &packageHead, sizeof(PackageHeader));
- memcpy(frameBuffer+sizeof(PackageHeader), dataBuffer+packageHead.uDataPackageOffset, UDP_MAX_SIZE);
- int length=udpsocketSend->writeDatagram(
- (const char*)frameBuffer, packageHead.uTransPackageSize,
- QHostAddress(ui->lineEditIP->text()), ui->lineEditPort->text().toInt());
- if(length!=packageHead.uTransPackageSize)
- {
- qDebug()<<"Failed to send image";
- }
- currentPacketIndex++;
- }
- else
- {
- packageHead.uTransPackageSize = sizeof(PackageHeader)+(dataLength-currentPacketIndex*UDP_MAX_SIZE);
- packageHead.uDataPackageCurrIndex = currentPacketIndex+1;
- packageHead.uDataPackageOffset = currentPacketIndex*UDP_MAX_SIZE;
- memcpy(frameBuffer, &packageHead, sizeof(PackageHeader));
- memcpy(frameBuffer+sizeof(PackageHeader), dataBuffer+packageHead.uDataPackageOffset, dataLength-currentPacketIndex*UDP_MAX_SIZE);
- int length=udpsocketSend->writeDatagram(
- (const char*)frameBuffer, packageHead.uTransPackageSize,
- QHostAddress(ui->lineEditIP->text()), ui->lineEditPort->text().toInt());
- if(length!=packageHead.uTransPackageSize)
- {
- qDebug()<<"Failed to send image";
- }
- currentPacketIndex++;
- }
- }
先将图片转换为QBuffer,这样就可以获取图片的数据和长度,然后根据指定大小UDP_MAX_SIZE分包。在包头中设置每包数据的特征,然后将包体加到包头后面发送出去。
2.组包
- static int num = 1;
- static uint size = 0;
- PackageHeader *packageHead = (PackageHeader *)datagram.data();
- if (packageHead->uDataPackageCurrIndex == num)
- {
- num++;
- size += packageHead->uTransPackageSize-packageHead->uTransPackageHdrSize;
- if (size > 1024*1000)
- {
- qDebug()<<"image too big";
- num = 1;
- size = 0;
- return;
- }
- if (packageHead->uDataPackageOffset > 1024*1000)
- {
- qDebug()<<"image too big";
- packageHead->uDataPackageOffset = 0;
- num = 1;
- size = 0;
- return;
- }
- memcpy(imageData.data+packageHead->uDataPackageOffset, datagram.data()+packageHead->uTransPackageHdrSize,
- packageHead->uTransPackageSize-packageHead->uTransPackageHdrSize);
- if ((packageHead->uDataPackageNum == packageHead->uDataPackageCurrIndex)
- && (size == packageHead->uDataSize))
- {
- imageData.length = packageHead->uDataSize;
- QImage image;
- image.loadFromData((uchar *)imageData.data,imageData.length,"JPG");
- QPixmap pixmap=QPixmap::fromImage(image);
- ui->labelImage_2->setPixmap(pixmap);
- recvImageNum++;
- ui->lineEditRevFrame->setText(QString::number(recvImageNum));
- ui->lineEditRevSize->setText(QString::number(imageData.length));
- memset(&imageData,0,sizeof(UdpUnpackData));
- num = 1;
- size = 0;
- }
- }
- else
- {
- num = 1;
- size = 0;
- memset(&imageData,0,sizeof(UdpUnpackData));
- }
组包是分包的逆过程,根据包头中“数据的总大小”和“数据被分成包的个数”两个字段可以判断组包是否完整,如果完整就在接收端显示出来。
三.示例
界面包括发送端和接收端,接收端的IP地址是自动获取的本机IP地址,上图将发送端的IP地址设置为与接收端IP地址相同,可实现自发自收。
每秒帧数可实时设置每秒发送图片的张数,发送帧数表示一共发送了多少张,接收帧数表示一共接收了多少张,图片大小表示每张图片多少Byte。
可将示例运行于两台计算机,实现双向收发。
源码链接:见http://blog.csdn.net/caoshangpa/article/details/52681572的评论
http://blog.csdn.net/caoshangpa/article/details/52681572
Qt通过UDP传图片,实现自定义分包和组包的更多相关文章
- 异步SOCKET分包和组包的一种通用算法
unit uPackage;// 应用协议// cxg 2016-9-23// 包=包头+包体 interface uses SysUtils, Classes, PeachCtrl.Net.Iocp ...
- Socket之UDP分包组包
一般传输大的文件和信息的时候需要涉及到分包和组包,方法有很多,下面一种是借鉴了别人的思路,供大家参考哈 分包 1.取出需要传输的文件和字符的长度和大小放入缓存区里面: 2.设定固定传输的长度,用需要传 ...
- 使用 Qt 获取 UDP 数据并显示成图片
一个项目,要接收 UDP 数据包,解析并获取其中的数据,主要根据解析出来的行号和序号将数据拼接起来,然后将拼接起来的数据(最重要的数据是 R.G.B 三个通道的像素值)显示在窗口中.考虑到每秒钟要接收 ...
- 使用 Qt 获取 UDP 数据并显示成图片(2)
本文首发于 BriFuture 的 个人博客 在我的前一篇文章 使用 Qt 获取 UDP 数据并显示成图片 中,我讲了如何用 Python 模拟发送数据,如何在 Qt 中高效的接收 UDP 数据包并将 ...
- 7.Vue-Quill-Editor图片插入自定义
Vue-Quill-Editor图片插入自定义 前言: 因为在项目中前端采用了Vue来实现,正好用到了富文本编辑器这一块,于是,经过技术上的选择,决定使用Vue-Quill-Editor. 使用的过程 ...
- Qt自适应大小显示图片,添加菜单
由于后面的图像处理需要UI,OpenCV自带也不怎么会,MFC实在懒得学的.听同学说Qt不错,就用Qt做UI了. 本文主要介绍三个内容:在Qt Creator中使用OpenCV2.Qt中自适应显示图片 ...
- iOS开发——UI进阶篇(十八)核心动画小例子,转盘(裁剪图片、自定义按钮、旋转)图片折叠、音量震动条、倒影、粒子效果
一.转盘(裁剪图片.自定义按钮.旋转) 1.裁剪图片 将一张大图片裁剪为多张 // CGImageCreateWithImageInRect:用来裁剪图片 // image:需要裁剪的图片 // re ...
- 用qt代码怎样编写图片保存格式[qt4.6]
用qt代码怎样编写图片保存格式 qt提供了多个保存图片的接口,比较常用的接口如下 bool QPixmap::save ( const QString & fileName, const ch ...
- iOS-AFNetworking-上传图片
一.AFNetworking上传图片 /** 需要Demo,可留言 */ 1.利用AFnetworking上传图片. 2.注意将UIImage类型转化为NSData类型,传给后台. 二.Demo // ...
随机推荐
- OpenCV检测人脸实例代码
下面是使用OpenCV通过在硬盘中读入图像来对其进行Haar人脸检测的代码. //包含头文件 #include <opencv2/core/core.hpp> #include " ...
- android中 回调方法,怎么转变为阻塞执行的方法
项目中需要用到在wifi这一块,扫描附近wifi,然后一个个遍历所有wifi,并且尝试连接,所以就在这里需要每连接一个wifi进行阻塞,当连接失败的时候才进行尝试下一个连接,当连接成功时则break. ...
- android 怎么动态设置button 的style
网上找了很多,还是没有直接的解决办法,button没有setstyle这个方法.因此我的解决办法如下: 直接动态设置各个属性 Button themeBtn = new Button(this); t ...
- Oracle VM VirtualBox 5.0 CentOS 6.4 共享文件夹
首先在主机(win7)的硬盘建立需要共享文件夹 例如 D:\share_test 然后虚拟机光驱加载Oracle VM VirtualBox安装目录的iso C:\Program Files\Ora ...
- Servlet主要的作用
1,收集Request传递过来的参数: 2,把这些参数组织成模型需要的类型: 3,调用模型进行逻辑功能处理: 4,选择下一个页面,先准备好一个页面需要的数据,然后转向下一个页面.
- MySQL中求年龄
时间函数: 1.curdate() --- 当前系统日期 调取: select curdate() 2.curtime() --- 当前系统时间 调取: select curtime() 3.now( ...
- SPFA 原理剖析代码实现分析比较
算法简介 SPFA(Shortest Path Faster Algorithm)是Bellman-Ford算法的一种队列实现,减少了不必要的冗余计算. 算法流程 算法大致流程是用一个队列来进行维护. ...
- 请实现一个函数用来找出字符流中第一个只出现一次的字符。例如,当从字符流中只读出前两个字符"go"时,第一个只出现一次的字符是"g"。当从该字符流中读出前六个字符“google"时,第一个只出现一次的字符是"l"。
// test20.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include<iostream> #include< ...
- SPOJ 057 Supernumbers in a permutation
原题链接:http://www.spoj.com/problems/SUPPER/ 这道题n<=200000,那么确定为nlogn的算法,再定位到求LIS的O(nlogn)的算法. 对于每个a[ ...
- 【BZOJ】【3550】【ONTAK2010】Vacation
网络流/费用流 Orz太神犇了这题…… 我一开始想成跟Intervals那题一样了……每个数a[i]相当于覆盖了(a[i]-n,a[i]+n)这个区间……但是这样是错的!!随便就找出反例了……我居然还 ...