Qt5--UDP图片传输并显示

利用UDP发送图片是试验UDP传输视频流的基础。


UDP发送端
UDPSender.pro

#-------------------------------------------------
#
# Project created by QtCreator 2017-04-06T20:01:35
#
#-------------------------------------------------

QT       += core gui
QT       += network
greaterThan(QT_MAJOR_VERSION, 4): QT += widgets

TARGET = UDPSender
TEMPLATE = app


SOURCES += main.cpp\
        mainwindow.cpp

HEADERS  += mainwindow.h

FORMS    += mainwindow.ui

mainwindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include <QUdpSocket>
#include <QFile>


#define TRAN_SIZE 1024;
struct ImageFrameHead {
    int funCode;                        //!功能码
    unsigned int uTransFrameHdrSize;    //!sizeof(WIFI_FRAME_HEADER)
    unsigned int uTransFrameSize;       //!sizeof(WIFI_FRAME_HEADER) + Data Size

    //数据帧变量
    unsigned int uDataFrameSize;        //数据帧的总大小
    unsigned int uDataFrameTotal;       //一帧数据被分成传输帧的个数
    unsigned int uDataFrameCurr;        //数据帧当前的帧号
    unsigned int uDataInFrameOffset;    //数据帧在整帧的偏移
};
namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();


private slots:
    void on_pushButton_clicked();

private:
    Ui::MainWindow *ui;
    QUdpSocket *m_udpSocket;
};
#endif // MAINWINDOW_H

mainwindow.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QTime>
#include <QDesktopWidget>

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    m_udpSocket = new QUdpSocket(this);
}

MainWindow::~MainWindow()
{
    delete ui;
}

void MainWindow::on_pushButton_clicked()
{
    while(1) {
//        QPixmap result = QPixmap();
//        result = QPixmap::grabWindow(QApplication::desktop()->winId(), 0, 0, 800, 600); //抓取当前屏幕的图片

//        result.save("D:\\LX\\Work\\Project\\802PC\\TcpServetTest\\802TCPServer\\eee.jpg");
        QFile file("D:\\tempdata.jpg");
        if (!file.open(QIODevice::ReadOnly))
            return;
        char *m_sendBuf = new char[1024];

        int size = file.size();
        int num = 0;
        int count = 0;
        int endSize = size%996;
        if (endSize == 0) {
            num = size/996;
        }
        else {
            num = size/996+1;
        }


        while (count < num) {
            memset(m_sendBuf, 0, 1024);

            ImageFrameHead mes;
            mes.funCode = 24;
            mes.uTransFrameHdrSize = sizeof(ImageFrameHead);
            if ((count+1) != num) {
                mes.uTransFrameSize = 996;
            }
            else {
                mes.uTransFrameSize = endSize;
            }
            //qDebug()<<size;
            mes.uDataFrameSize = size;
            mes.uDataFrameTotal = num;
            mes.uDataFrameCurr = count+1;
            mes.uDataInFrameOffset = count*(1024 - sizeof(ImageFrameHead));

            file.read(m_sendBuf+sizeof(ImageFrameHead), 1024-sizeof(ImageFrameHead));

            memcpy(m_sendBuf, (char *)&mes, sizeof(ImageFrameHead));
            m_udpSocket->writeDatagram(m_sendBuf, mes.uTransFrameSize+mes.uTransFrameHdrSize, QHostAddress("192.168.56.1"), 65522);
            QTime dieTime = QTime::currentTime().addMSecs(1);
            while( QTime::currentTime() < dieTime )
                QCoreApplication::processEvents(QEventLoop::AllEvents, 100);
            count++;
        }
        file.close();
        QTime dieTime = QTime::currentTime().addMSecs(10);
        while( QTime::currentTime() < dieTime )
            QCoreApplication::processEvents(QEventLoop::AllEvents, 100);
        //qDebug("ok");
    }

}

main.cpp

#include "mainwindow.h"
#include <QApplication>

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    MainWindow w;
    w.show();

    return a.exec();
}

ui
这里写图片描述
UDP接收端
UDPReceiver.pro

#-------------------------------------------------
#
# Project created by QtCreator 2017-04-06T20:10:32
#
#-------------------------------------------------

QT       += core gui
QT       += network
greaterThan(QT_MAJOR_VERSION, 4): QT += widgets

TARGET = UDPReceiver
TEMPLATE = app


SOURCES += main.cpp\
        mainwindow.cpp \
    udpthread.cpp

HEADERS  += mainwindow.h \
    udpthread.h

FORMS    += mainwindow.ui

mainwindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QMainWindow>
#include "udpthread.h"

namespace Ui {
class MainWindow;
}

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();
private slots:
    void slotRecv(char *buf, int len);

private:
    Ui::MainWindow *ui;
    UdpThread *m_udpThread;
};

#endif // MAINWINDOW_H

udpthread.h

#ifndef UDPTHREAD_H
#define UDPTHREAD_H

#include <QThread>
#include <QUdpSocket>
struct ImageFrameHead {
    int funCode;                        //!功能码
    unsigned int uTransFrameHdrSize;    //!sizeof(WIFI_FRAME_HEADER)
    unsigned int uTransFrameSize;       //!sizeof(WIFI_FRAME_HEADER) + Data Size

    //数据帧变量
    unsigned int uDataFrameSize;        //数据帧的总大小
    unsigned int uDataFrameTotal;       //一帧数据被分成传输帧的个数
    unsigned int uDataFrameCurr;        //数据帧当前的帧号
    unsigned int uDataInFrameOffset;    //数据帧在整帧的偏移
};

class UdpThread : public QThread
{
    Q_OBJECT
public:
    explicit UdpThread(QObject *parent = 0);
    ~UdpThread();

protected:
    virtual void run();

signals:
    void sigRecvOk(char *buf, int len);

public slots:
    void slotRecv();
private:
    QUdpSocket *m_udpSocket;
    char *m_buf;

};

#endif // UDPTHREAD_H

mainwindow.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    m_udpThread = new UdpThread;
    connect(m_udpThread, SIGNAL(sigRecvOk(char*,int)), this, SLOT(slotRecv(char*,int)));
    m_udpThread->start();
    qDebug("OK");
}

MainWindow::~MainWindow()
{
    delete ui;
}
void MainWindow::slotRecv(char * buf, int len)
{

    QPixmap pixmap;
    pixmap.loadFromData((uchar*)buf, len, "JPG");
    ui->label->setPixmap(pixmap);

}

udpthread.cpp

#include "udpthread.h"

UdpThread::UdpThread(QObject *parent) :
    QThread(parent)
{
    m_buf = new char[1024*1024];
    memset(m_buf, 0, 1024);

}

UdpThread::~UdpThread()
{
    m_udpSocket->close();
    delete m_buf;
}

void UdpThread::run()
{
    m_udpSocket = new QUdpSocket;
    connect(m_udpSocket, SIGNAL(readyRead()), this, SLOT(slotRecv()),Qt::DirectConnection);
    qDebug()<< m_udpSocket->bind(QHostAddress::Any, 65522);
    exec();
}

void UdpThread::slotRecv()
{
    char *recvBuf = new char[1052];
    memset(recvBuf, 0, 1052);
    //qDebug("start");
    while(m_udpSocket->hasPendingDatagrams()) {
        memset(recvBuf, 0, 1052);
        int size = m_udpSocket->pendingDatagramSize();

        //qDebug()<<"size"<<size;
        m_udpSocket->readDatagram(recvBuf, size);
        ImageFrameHead *mes = (ImageFrameHead *)recvBuf;
        //qDebug()<<mes->uDataFrameCurr<<mes->uDataFrameTotal;
        if (mes->funCode == 24) {
            memcpy(m_buf+mes->uDataInFrameOffset, (recvBuf+ sizeof(ImageFrameHead)), mes->uTransFrameSize);
            if (mes->uDataFrameCurr == mes->uDataFrameTotal) {
                qDebug("write");
                emit sigRecvOk(m_buf, mes->uDataFrameSize);
            }
        }

    }
    delete recvBuf;
}

main.cpp

#include "mainwindow.h"
#include <QApplication>

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    MainWindow w;
    w.show();

    return a.exec();
}

ui
这里写图片描述

测试:
这里写图片描述

  • 5
    点赞
  • 46
    收藏
    觉得还不错? 一键收藏
  • 7
    评论
### 回答1: 基于Qt平台利用UDP协议传输图片的方法如下: 1. 双方共同约定数据传输格式:在发送端和接收端都需要确定图片的格式,比如JPEG、PNG等。这样可以确保双方在传输过程中使用相同的格式解析图片数据。 2. 发送端将大图片切分为适当大小的数据包:由于UDP协议没有拆分和重组数据包的机制,所以发送端需要将大图片切分为适当大小的数据包以适应UDP协议的传输。 3. 发送端将切分后的数据包发送到接收端:利用UDP协议的广播或单播功能,发送端将切分后的数据包发送到接收端的指定地址和端口。 4. 接收端接收数据包并重新组装图片:接收端侦听指定的地址和端口,接收来自发送端的数据包。接收端需要根据数据包的顺序和大小重新组装图片。 5. 接收端显示接收到的图片:接收端在重新组装图片后,可以利用Qt平台提供的图形库显示接收到的图片。 需要注意的是,由于UDP协议是一种面向无连接并且不可靠的协议,因此在图片传输时可能会存在数据包丢失或乱序的情况。为了提高传输的可靠性,可以在应用层引入一些算法,比如给每个数据包添加序号、校验和等信息来检测和纠正错误。另外,还可以考虑引入一些窗口管理机制来控制发送和接收的速率,以适应网络环境的变化。 ### 回答2: 基于Qt平台利用UDP协议传输图片需要进行以下步骤: 首先,我们需要将图片进行分割成小块。由于UDP协议有传输数据大小的限制,一次传输整个大图片可能会导致数据丢失或者传输过程中的延迟较高。因此,我们需要将大图片分割成多个小块。 接下来,我们使用Qt提供的UDP套接字类(QUdpSocket)来实现UDP通信。我们创建一个UDP发送端和一个UDP接收端,通过网络传输图像的小块。 在发送端,我们将分割后的图像块转换为QByteArray类型,并使用QUdpSocket的writeDatagram()函数发送数据包。同时,我们可以设置目标IP地址和端口号,以便接收端能正确接收数据。 在接收端,我们创建一个QUdpSocket对象,并使用bind()函数将其绑定到指定的IP地址和端口号上。然后,我们使用QUdpSocket的readyRead()信号与对应的槽函数来接收数据包。在槽函数中,我们可以使用readDatagram()函数来读取接收到的数据,并进行相应的处理。 在接收端,我们还需要重组接收到的图像块。我们可以使用一个QByteArray来存储接收到的所有数据,并使用一个计数器(或其他方式)来确定是否已经接收到了所有图像块。一旦确定接收到了完整的图像数据,我们可以将其重新组合成完整的图像。 最后,我们可以通过显示图像窗口或保存图像文件来查看接收到的完整图像。 需要注意的是,由于UDP协议的不可靠性,传输过程中会存在丢包或乱序等情况。因此,我们需要进行适当的错误处理和容错机制,例如设置超时时间、重传机制等,以确保图像数据能够完整地传输和接收。 ### 回答3: 基于Qt平台利用UDP协议传输图片的步骤如下: 1. 创建一个Qt应用程序,包括发送端和接收端两个部分。 2. 在发送端,使用Qt的网络模块创建一个UDP套接字,并绑定到特定的IP地址和端口号上。 3. 将要发送的大图片分割成小块,以便每次发送一个固定大小的数据包。可以使用Qt的图像处理模块来切割图片。 4. 将每个小块图片的数据打包成一个UDP数据报文,其中包括数据包的序号和总数据包数等信息。 5. 在发送端,使用UDP套接字的writeDatagram函数将数据报文发送给接收端的IP地址和端口号。 6. 在接收端,创建一个UDP套接字,并绑定到相同的IP地址和端口号。 7. 在接收端,使用UDP套接字的readyRead信号连接一个槽函数,该槽函数在接收到一个数据包时被触发。 8. 在槽函数中,通过UDP套接字的readDatagram函数读取接收到的数据包。 9. 在接收端,将接收到的数据包中的图片数据存储起来。 10. 重复步骤7到9,直到接收到所有的数据包。 11. 在接收端,使用Qt的图像处理模块将接收到的图片数据合并成完整的大图片。 需要注意的是,UDP是一种不可靠的传输协议,因此在实际应用中需要考虑一些额外的因素,如数据包的丢失和乱序等情况的处理。可以通过引入一些重传机制和序号控制等方式来增加数据传输的可靠性。
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值