Django的form表单之文件上传

简介: 在生成input标签的时候可以指定input标签的类型为file类型 Title{{ error_message }} {% csrf_token %} 此时,在网页上页示如下如果网页上提交的是用户名和密码等,通过键值对发送到服务端。

在生成input标签的时候可以指定input标签的类型为file类型

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<h4>{{ error_message }}</h4>
<form action="/index/" method="post">
    {% csrf_token %}
    <p><input type="file" name="up_file"></p>
    <input type="submit">
</form>
</body>
</html>

此时,在网页上页示如下

img_8b16165e4f7f6adbaf27a872d505f5b2.png

如果网页上提交的是用户名和密码等,通过键值对发送到服务端。

一组键值代表一个标签及标签对应的值。

在网页上选择一张图片,并使用post方式提交,在服务端打印request.POST

    def index(request):
        if request.method=="POST":
            print(request.POST)
    
        return render(request,"index.html",locals())
    

打印的信息如下:

<QueryDict: {'csrfmiddlewaretoken': ['opmSmENIrgdGJJN'], 'up_file': ['1.png']}>

提交的文件名也在这个字典中,取出这个文件名

    def index(request):
        if request.method=="POST":
            print(request.POST.get("up_file"))
            print(type(request.POST.get("up_file")))
    
        return render(request,"index.html",locals())
    

打印结果如下:

1.png
<class 'str'>

想取出的是上传的文件,然而取出来的却是上传的文件的文件名

由此可知,上传的文件没有跟form表单提交的数据在一起

因为上传的文件通常大小比较大,所以Django默认会把上传的文件放在一个具体的文件夹中

打印request.FILES的信息

    def index(request):
        if request.method=="POST":
            print(request.POST.get("up_file"))
            print(type(request.POST.get("up_file")))
            print("files:",request.FILES)
    
        return render(request,"index.html",locals())

打印结果如下

1.png
<class 'str'>
files: <MultiValueDict: {}>

request.FILES打印的结果是一个空的字典,问题出在上传文件的方式上

由于上传文件时在客户端与服务端传输的是二进制数据,与字符串数据不一样。

传输二进制数据,不管是在form表单,还是在Ajax中,都有自己的传输方式。

在form表单中,上传文件时要使用分片传输的方式。

修改index.html文件

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<h4>{{ error_message }}</h4>
<form action="/index/" method="post" enctype="multipart/form-data">
    {% csrf_token %}
    <p><input type="file" name="up_file"></p>
    <input type="submit">
</form>
</body>
</html>

重新上传文件,在服务端打印信息如下

None
<class 'NoneType'>
files: <MultiValueDict: {'up_file': [<InMemoryUploadedFile: 1.png (image/png)>]}>

根据打印结果,request.FILES中可以看到上传的文件

打印结果是一个字典类型,字典的键是form表单中定义的标签的name属性值,而其值是所上传的文件的对象

打印上传文件的对象

def index(request):
    if request.method=="POST":

        print("files:",request.FILES.get("up_file"))
        print(type(request.FILES.get("up_file")))

    return render(request,"index.html",locals())

打印结果

files: 1.png
<class 'django.core.files.uploadedfile.InMemoryUploadedFile'>

结果显示所取得的文件的类型是一个在内存中的上传文件

获取上传文件在内存中的名字

def index(request):
    if request.method=="POST":

        print(type(request.FILES.get("up_file")))

        file_obj=request.FILES.get("up_file")

        print(file_obj.name)


    return render(request,"index.html",locals())

打印结果如下

<class 'django.core.files.uploadedfile.InMemoryUploadedFile'>
1.png

既然知道了文件在内存中的名字,就可以在服务端写入这个文件

def index(request):
    if request.method=="POST":
        file_obj=request.FILES.get("up_file")

        f1=open(file_obj.name,"wb")

        for i in file_obj.chunks():
            f1.write(i)

        f1.close()

    return render(request,"index.html",locals())

再次选择上传文件,提交后,就可以在服务端后台看到所上传的文件

可以在settings.py文件中设定上传文件的路径,或者在打开文件句柄的时候进行路径拼接来把上传的文件保存在指定的目录下

目录
相关文章
|
5月前
|
JSON 前端开发 API
Django 后端架构开发:通用表单视图、组件对接、验证机制和组件开发
Django 后端架构开发:通用表单视图、组件对接、验证机制和组件开发
75 2
|
5月前
|
数据采集 Python
Django 表单
【8月更文挑战第24天】
37 3
|
5月前
|
Python
Django表单组件
【8月更文挑战第20天】
43 1
|
5月前
|
前端开发 JavaScript 数据处理
Django的表单处理
【8月更文挑战第16天】
29 2
|
7月前
|
Python
Django表单
【6月更文挑战第13天】Django表单。
37 5
|
7月前
|
测试技术 数据库 Python
使用django构建表单测试
【6月更文挑战第14天】该文档介绍了如何对本地库进行自动化测试,特别是关注于代码结构和模型测试。作者鼓励为其他模型和表单创建类似的测试,并提及测试应避免对底层框架的重复验证。
76 0
使用django构建表单测试
|
7月前
|
Python
Django表单
【6月更文挑战第3天】Django表单。
35 1
|
7月前
|
Python
Django表单
【6月更文挑战第3天】Django表单。
29 1
|
8月前
|
JSON 数据库 数据格式
Django之Form组件
Django之Form组件
|
8月前
|
数据库 开发者 UED
如何使用Django的Form组件
如何使用Django的Form组件
91 0