本文使用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. 【29.89%】【codeforces 734D】Anton and Chess

    time limit per test4 seconds memory limit per test256 megabytes inputstandard input outputstandard o ...

  2. VS Code插件之Cordova Tools

    原文:VS Code插件之Cordova Tools 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/u011127019/article/detai ...

  3. 构建工具Gradle

    1.Summary   从Android团队开始宣布放弃Eclipse转投Android Studio时,构建工具Gradle进入了Android开发者的视野.而随着热修复.插件化.编译时注解的流行, ...

  4. C++基础之字符数组和字符串

    无意中发现了一个非常有意思的技术类型小品文系列,通过大牛指导菜鸟的方式,解说讲C++知识,有的非常基础却是开发中easy忽略的地方. [Elminster的专栏] http://blog.csdn.n ...

  5. React中的state与props的再理解

    props可以看做是 property 的缩写的复数,可以翻译为属性,类似于HTML 标签的自定义属性.在大多数React教程里讲 state 和 props 主要的区别在于 props 是不可变的, ...

  6. 【21.67%】【codeforces 727B】Bill Total Value

    time limit per test1 second memory limit per test256 megabytes inputstandard input outputstandard ou ...

  7. webpack优化经验1(持续)

    1 不知道该优化哪里 先开启gzip压缩,这样可以很直接的减少请求包的体积,效果显著,不过需要在服务器端作相应的配置才能生效 2拆分vendor包, 减少单体包的体积,并行加载 通过配置,将不同的公用 ...

  8. 设置vista和win7进入Debug模式

    转载请标明是引用于 http://blog.csdn.net/chenyujing1234 欢迎大家拍砖 设置vista和win7进入Debug模式: 1. bcdedit /copy  {curre ...

  9. 学习 NLP(一)—— TF-IDF

    TF-IDF(Term Frequency & Inverse Document Frequency),是一种用于信息检索与数据挖掘的常用加权技术.它的主要思想是:如果某个词或短语在一篇文章中 ...

  10. Python中 如何将一个字符串分成一个个字符

    其实   一个字符串  实质也是 一个列表 就很简单了: a = ' for item in a: print(item) 打印结果: 121512 如果进而要统计字符出现的次数 , 那就很简单了.