本文使用Qt Creator用HTTP POST的方法上传文件,并给出一个上传文件的例程。

本文主要客户端,所以对于服务器端程序编写的描述会比较简略

服务器使用Django编写,django服务器接收文件的方法在文章http://www.cnblogs.com/fnng/p/3740274.html中有较为清晰的讲解,我搭建的服务器端程序除了没有网页客户端以及部分变量名称不同以外,基本上与这篇文章的服务器搭建过程一样。

如果服务器端程序发生变化,这篇文章后面给出的客户端例程可能就不再适用。因此如果运行客户端程序之后发现服务器端不能接收到文件,不要直接认为这篇文章给出的客户端例程是错误的,也可能是其他问题导致的。

在制作程序时,我并没有尝试上传大文件、同时上传多个文件以及文件名中包含中文这三种情况。因此在这三种情况下程序很有可能出现BUG。

示例程序链接:http://pan.baidu.com/s/1i5NWsHR

1.      服务器端

服务器端程序基本参照http://www.cnblogs.com/fnng/p/3740274.html。这里直接贴出代码,搭建过程以及代码描述等就不进行说明了。

disk/views.py:

from django.shortcuts import render,render_to_response

from django.http import HttpResponse

from django import forms

from .models import User

import os

# Create your views here.

class UserForm(forms.Form):

username=forms.CharField()

upload_file=forms.FileField()

def index(request):

if request.method=="POST":

uf=UserForm(request.POST,request.FILES)

print(str(request.POST))

print(str(request.FILES))

if uf.is_valid():

username=uf.cleaned_data['username']

upload_file=uf.cleaned_data['upload_file']

dir_name='./upload/'+username

if os.path.exists(dir_name)==False:

os.mkdir(dir_name)

user=User()

user.username=username

user.upload_file=upload_file

user.save()

return HttpResponse('upload ok!')

else:

uf=UserForm()

return HttpResponse('000!')

disk/models.py:

from django.db import models

# Create your models here.

def content_file_name(instance,filename):

s='/'.join(['upload',instance.username,filename])

return s

class User(models.Model):

username=models.CharField(max_length=30)

upload_file=models.FileField(upload_to=content_file_name)

def __unicode__(self):

return self.username

disk/urls.py:

from django.conf.urls import url

from . import views

urlpatterns=[

#View the information of user.

url(r'^$',views.index,name='index'),

]

2.         客户端程序应该上传的数据

客户端程序在向服务器端上传文件时,不能只上传文件的内容。如果上传的数据中只有文件的内容,那么服务器端是不能正常接收到文件的。客户端上传的内容应该包括普通的表单数据、文件头部以及文件内容三大部分,并对需要传输的数据进行规范化地拼接,按照特定的格式上传数据。

对于我用django制作的这个服务器端程序来说,客户端需要上传一个name为username的普通数据,以及一个name为upload_file的文件数据。下面的代码将需要发送的数据进行拼接并使用POST请求将数据发送给服务器。

3.         新建工程过程中需要注意的地方

在pro文件中要加上QT += network 如果不添加会报错

在代码文件的最前面添加包含 #include 。(不添加的话程序会编译失败)我自己在学习使用Qt进行HTTP通信时发现网上的很多文章在贴出代码时并没有贴出前面的#include部分,导致我花了很长时间去找需要引用的文件。

4.         客户端上传文件的代码

void UpLoadForm(QString Path,QMap<</span>QString,QString> params,QStringfileFormName,QFile *uploadFile,QString newFileName){

QString BOUNDARY=QUuid::createUuid().toString();

QByteArray sb=QByteArray();

//先上传普通的表单数据

for(QMap<</span>QString,QString>::Iteratort=params.begin();t!=params.end();t++){

sb.append("--"+BOUNDARY+"\r\n");

sb.append(QString("Content-Disposition: form-data;name=\"")+t.key()+QString("\"")+QString("\r\n"));

sb.append("\r\n");

sb.append(t.value()+"\r\n");

}

//上传文件的头部

sb.append("--"+BOUNDARY+"\r\n");

sb.append(QString("Content-Disposition: form-data;name=\"")+fileFormName+QString("\";filename=\"")+newFileName+QString("\"")+QString("\r\n"));

sb.append("\r\n");

//上传文件内容

if(!uploadFile->open(QIODevice::ReadOnly)){

return;

}

sb.append(uploadFile->readAll());

sb.append("\r\n");

sb.append("--"+BOUNDARY+"\r\n");

//编辑HTTP头部

QNetworkAccessManager *_uploadManager=new QNetworkAccessManager();

QNetworkRequest request=QNetworkRequest(QUrl(Path));

request.setRawHeader(QString("Content-Type").toLatin1(),QString("multipart/form-data;boundary="+BOUNDARY).toLatin1());

request.setRawHeader(QString("Content-Length").toLatin1(),QString::number(sb.length()).toLatin1());

//执行post请求

_uploadManager->post(request,sb);

}

5.         编写main函数调用前面上传文件的代码,完成测试。

在这里,我们提交了一个name为username的普通表单数据,它的值为10005。我们同时还提交了一个name为upload_file的文件数据。我们上传了本地的”:/Untitled.png”文件,上传成功之后服务器会接收到这个文件,保存时命名为”1A.png”

int main(int argc, char *argv[])

{

QCoreApplication a(argc, argv);

QString path="http://192.168.252.2:8000/disk/"; //服务器的url

QMap<</span>QString,QString> params_send; //上传的普通参数 在本程序中 需要上传一个普通参数为"username"

params_send.insert("username","10005");

QString fileFormName="upload_file"; //上传文件表单中的名字

QFile *file=new QFile(":/Untitled.png");

QString newFileName="1A.png";

UpLoadForm(path,params_send,fileFormName,file,newFileName);

return a.exec();

}

6.         效果

我们把这张图片”untitled.png”上传给服务器

服务器接收到了客户端发来的图片,表单数据中username=10005,所以新建了一个文件夹”10005”,将接收到的图片保存在了这个文件夹中并命名为”1A.png”

7.         参考资料

http://www.cnblogs.com/fnng/p/3740274.html

https://forum.qt.io/topic/11086/solved-qnetworkaccessmanager-uploading-files

http://blog.csdn.net/lmj623565791/article/details/23781773

如果发现问题欢迎在下面留言或私信我

http://blog.sina.com.cn/s/blog_15d207b300102xvqz.html

Qt通过HTTP POST上传文件(python做服务端,附下载)的更多相关文章

  1. react native android 上传文件,Nodejs服务端获取上传的文件

    React Native端 使用react-native-image-picker 做出选择图片的操作,选择完成后,直接将图片Post至服务器,保存在服务器的某个地方(保存图片的路径需要公开显示),并 ...

  2. ASP.NET Core单文件和多文件上传并保存到服务端

    前言: 在我们日常开发中,关于图片,视频,音频,文档等相关文件上传并保存到服务端中是非常常见的一个功能,今天主要是把自己在开发中常用的两种方式记录下来方便一下直接使用,并且希望能够帮助到有需要的同学! ...

  3. linux下远程服务器批量执行命令及SFTP上传文件 -- python实现

    之前写过一个python远程执行命令的脚本,但在一个性能测试中,要将程序批量分发到不同服务器,程序无法使用,再将之前的脚本更新,加入批量上传的功能.之前脚本地址:http://www.cnblogs. ...

  4. SpringMVC上传文件以流方式判断类型附常用类型

    // 此类中判断类型所截取的byte 长度暂不确定,请使用者测试过使用 package com.tg.common.other; import com.tg.common.tginterface.TG ...

  5. C# 向服务器上传文件(客服端winform、服务端web)

    转载 首先写客服端,winform模拟一个post提交: /// <summary> /// 将本地文件上传到指定的服务器(HttpWebRequest方法) /// </summa ...

  6. 科普文:Node.js 如何上传文件到后端服务【转】

    原文链接 https://www.yuque.com/egg/nodejs/httpclient-upload 背景 互联网时代,无数服务是基于 HTTP 协议进行通信的. 除了常见的 前端浏览器 - ...

  7. 重新想象 Windows 8.1 Store Apps (89) - 通信的新特性: 下载数据, 上传数据, 上传文件

    [源码下载] 重新想象 Windows 8.1 Store Apps (89) - 通信的新特性: 下载数据, 上传数据, 上传文件 作者:webabcd 介绍重新想象 Windows 8.1 Sto ...

  8. 前端AngularJS后端ASP.NET Web API上传文件

    本篇体验使用AngularJS向后端ASP.NET API控制器上传文件.    首先服务端: public class FilesController : ApiController { //usi ...

  9. python网络编程--FTP上传文件示例

    1.基础版(供学习了解原理使用,low) server服务端 import socket import struct import json server = socket.socket() ip_p ...

随机推荐

  1. hbase 从hbase上读取数据写入到hdfs

    Mapper package cn.hbase.mapreduce.hb2hdfs; import java.io.IOException; import org.apache.hadoop.hbas ...

  2. PCI GXL学习之二次开发篇

    作者:朱金灿 来源:http://blog.csdn.net/clever101 gxl的二次开发分初级和高级之分.初级是gxl提供了几百个模块供你编排成不同的作业.高级就是你可以编写你的算法模块,然 ...

  3. 51系列小型操作系统精髓 简单实现6 C语言版待改进

    #include "STC12C5A.H" #define TIMER_RELOAD()  {TL0=0x00;TH0=0xC4;}//使能T/C  初始10ms #define ...

  4. Gamma 函数及其应用

    1. Γ(⋅) 函数定义 Γ(α)=∫∞0tα−1e−tdt 可知以下基本性质: Γ(α+1)=αΓ(α)(分部积分法) Γ(1)=1 ⇒ Γ(n+1)=n! Γ(12)=π√ 2. 常见变形 对于 ...

  5. hudson绑定svn和vs2008实现持续构建

    作者:朱金灿 来源:http://blog.csdn.net/clever101 首先需要在服务器上安装以下工具: (1)hudson,我推荐从http://hudson-ci.org/downloa ...

  6. Android系统联系人全特效实现(上),分组导航和挤压动画

    记得在我刚接触Android的时候对系统联系人中的特效很感兴趣,它会根据手机中联系人姓氏的首字母进行分组,并在界面的最顶端始终显示一个当前的分组.如下图所示: 最让我感兴趣的是,当后一个分组和前一个分 ...

  7. WPF MVVM系列文章

    网上搜到了MSDN Magazine上Laurent Bugnion的系列文章. 以下为关于WPF的优秀实践,很有必要阅读. 很吸引人的标题有: IOC Containers and MVVM Mes ...

  8. 《北京IT报道》你可以成为下一个《万万没有想到》?

    10一个月29当天上午,一位名为<北京IT报道>在视频短剧IT朋友谁快速刷新的互联网商业圈.这种制作粗糙.演员表情僵硬.安置赤裸网络剧,但对布局和内容因IT互联网从业者的生活,深受广大用户 ...

  9. PBS 作业调度应用

    PBS(Portable Batch System),最初由 NASA 的 Ames 研究中心开发,主要为了提供一个能满足异构计算网络需要的软件包,用于灵活的批处理(Portable Batch Pr ...

  10. OpenGL(二十二) gluBuild2DMipmaps 加载Mip纹理贴图

    当纹理被用于渲染一个面积比它本身小很多的对象时,会由于纹理图像的降采样率不足而导致混叠现象,主要的表现特征是纹理图像的闪烁,出现纹理躁动.特别是在场景远近移动变换时,这种闪烁情况更为明显,严重可能会影 ...