Qt之QNetworkAccessManager 从本地和内存中上传数据到Http服务器

本文介绍了如何使用Qt的QNetworkAccessManager从本地和内存中上传数据到Http服务器。在本地上传时,需正确设置ContentTypeHeader和ContentDispositionHeader,并确保QFile的生命周期管理。内存上传则需将数据转换为QIODevice类型。注意QHttpMultiPart的删除管理,通常将其设置为QNetworkReply的子类。附带源码下载链接。
摘要由CSDN通过智能技术生成

简述

        接连做了好几个服务器的项目,例如文件传输用的Ftp和对象存储服务器(Object Storage Service),简单的信息传输用的WebServer,之前也有用过HttpServer不过都和WebServer一样简单的调用接口提交数据并没有上传过文件,正好趁这次案例有机会做了个测试实例;

效果

从本地上传

        从本地中读取一个图片上传,图片的ContentTypeHeader头部信息要填写和图片格式一致的类型,例如:jpg对应"image/jpg",bmp对应"image/bmp",png对应"image/png";ContentDispositionHeader头部信息的name要对应"file";QFile需要动态创建,因为要上传的文件可能很大,如果是局部的话函数退出就会销毁内存,导致上传失误甚至程序奔溃;最后因为QFile是动态创建,所以需要给其设置一个父类QHttpMultiPart,当父类被删除时也删除QFile;

void MainWindow::on_pbFromLocal_clicked()
{
    obtainData();

    // 文件路径
    QFile *file = new QFile("C:/Users/ilson/Desktop/HttpFileUp/logo.jpg");
    if (!file->open(QFile::ReadOnly)) {
        statusBar()->showMessage("file open failed");
        return;
    }

    // 设置请求
    QHttpPart imagePart;
    imagePart.setHeader(QNetworkRequest::ContentTypeHeader, QVariant("image/jpg"));
    imagePart.setHeader(QNetworkRequest::ContentDispositionHeader, QVariant("form-data; name=\"file\";filename=\"test.jpg\""));
    imagePart.setBodyDevice(file);

    QHttpPart codePart;
    codePart.setHeader(QNetworkRequest::ContentDispositionHeader, QVariant("form-data; name=\"productCode\""));
    codePart.setBody("code");

    QHttpPart addressPart;
    addressPart.setHeader(QNetworkRequest::ContentDispositionHeader, QVariant("form-data; name=\"macAddress\""));
    addressPart.setBody("A1:B2:C3:D4:E5:G6");

    //
    QHttpMultiPart *multiPart = new QHttpMultiPart(QHttpMultiPart::FormDataType);
    multiPart->append(imagePart);
    multiPart->append(codePart);
    multiPart->append(addressPart);
    file->setParent(multiPart);  // 用 multiPart 删除 file

    QNetworkReply *reply = m_http.post(QNetworkRequest(makeUrl()), multiPart);
    multiPart->setParent(reply); // 用 reply 删除 multiPart
    connect(reply, SIGNAL(finished()), SLOT(replyFinished()));
}

从内存上传

        从内存上传和从本地的区别就是数据需要重新保存成QIODevice类型;

// 压缩图像
// image     要压缩的图像
QByteArray MainWindow::compim(const QImage &image)
{
    QBuffer buffer;
    buffer.open(QIODevice::WriteOnly);
    image.save(&buffer, "jpg", 90);
    buffer.close();

    return buffer.data();
}

void MainWindow::on_pbFromMemory_clicked()
{
    obtainData();

    // 文件路径
    QImage image("C:/Users/ilson/Desktop/111.png");

    // 设置请求
    QHttpPart imagePart;
    imagePart.setHeader(QNetworkRequest::ContentTypeHeader, QVariant("image/png"));
    imagePart.setHeader(QNetworkRequest::ContentDispositionHeader, QVariant("form-data; name=\"file\";filename=\"test.png\""));
    imagePart.setBody(compim(image));

    QHttpPart codePart;
    codePart.setHeader(QNetworkRequest::ContentDispositionHeader, QVariant("form-data; name=\"productCode\""));
    codePart.setBody("code");

    QHttpPart addressPart;
    addressPart.setHeader(QNetworkRequest::ContentDispositionHeader, QVariant("form-data; name=\"macAddress\""));
    addressPart.setBody("A1:B2:C3:D4:E5:G6");

    //
    QHttpMultiPart *multiPart = new QHttpMultiPart(QHttpMultiPart::FormDataType);
    multiPart->append(imagePart);
    multiPart->append(codePart);
    multiPart->append(addressPart);

    QNetworkReply *reply = m_http.post(QNetworkRequest(makeUrl()), multiPart);
    multiPart->setParent(reply); // 用 reply 删除 multiPart
    connect(reply, SIGNAL(finished()), SLOT(replyFinished()));
}

注意 

        QHttpMultiPart的删除需要给其设置父类QNetworkReply,当QNetworkReply被删除时其子类也自会删除;

其他

        源码下载(折扣):https://item.taobao.com/item.htm?ft=t&id=756129976580

  • 10
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Ilson_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值