django 1.8 官方文档翻译: 3-3-1 文件上传
文件上传
当Django在处理文件上传的时候,文件数据被保存在request. FILES
(更多关于 request
对象的信息 请查看 请求和响应对象)。这篇文档阐述了文件如何上传到内存和硬盘,以及如何自定义默认的行为。
警告
允许任意用户上传文件是存在安全隐患的。更多细节请在用户上传的内容中查看有关安全指导的话题。
基本的文件上传
考虑一个简单的表单,它含有一个FileField
:
# In forms.py...
from django import forms
class UploadFileForm(forms.Form):title = forms.CharField(max_length=50)file = forms.FileField()
处理这个表单的视图会在request
中接受到上传文件的数据。FILES
是个字典,它包含每个FileField
的键 (或者 ImageField
,FileField
的子类)。这样的话就可以用request.FILES['file']
来存放表单中的这些数据了。
注意request.FILES
会仅仅包含数据,如果请求方法为POST
,并且发送请求的<form>
拥有enctype="multipart/form-data"
属性。否则request.FILES
为空。
大多数情况下,你会简单地从request
向表单中传递数据,就像绑定上传文件到表单描述的那样。这样类似于:
from django.http import HttpResponseRedirect
from django.shortcuts import render_to_response
from .forms import UploadFileForm# Imaginary function to handle an uploaded file.
from somewhere import handle_uploaded_filedef upload_file(request):if request.method == 'POST':form = UploadFileForm(request.POST, request.FILES)if form.is_valid():handle_uploaded_file(request.FILES['file'])return HttpResponseRedirect('/success/url/')else:form = UploadFileForm()return render_to_response('upload.html', {'form': form})
注意我们必须向表单的构造器中传递request.FILES
。这是文件数据绑定到表单的方法。
这里是一个普遍的方法,可能你会采用它来处理上传文件:
def handle_uploaded_file(f):with open('some/file/name.txt', 'wb+') as destination:for chunk in f.chunks():destination.write(chunk)
遍历UploadedFile.chunks()
,而不是使用read()
,能确保大文件并不会占用系统过多的内存。
UploadedFile
对象也拥有一些其他的方法和属性;完整参考请见UploadedFile。
使用模型处理上传文件
如果你在Model
上使用FileField
保存文件,使用ModelForm
可以让这个操作更加容易。调用form.save()
的时候,文件对象会保存在相应的FileField
的upload_to
参数指定的地方。
from django.http import HttpResponseRedirect
from django.shortcuts import render
from .forms import ModelFormWithFileFielddef upload_file(request):if request.method == 'POST':form = ModelFormWithFileField(request.POST, request.FILES)if form.is_valid():# file is savedform.save()return HttpResponseRedirect('/success/url/')else:form = ModelFormWithFileField()return render(request, 'upload.html', {'form': form})
如果你手动构造一个对象,你可以简单地把文件对象从request.FILE
赋值给模型:
from django.http import HttpResponseRedirect
from django.shortcuts import render
from .forms import UploadFileForm
from .models import ModelWithFileFielddef upload_file(request):if request.method == 'POST':form = UploadFileForm(request.POST, request.FILES)if form.is_valid():instance = ModelWithFileField(file_field=request.FILES['file'])instance.save()return HttpResponseRedirect('/success/url/')else:form = UploadFileForm()return render(request, 'upload.html', {'form': form})
上传处理器
当用户上传一个文件的时候,Django会把文件数据传递给上传处理器 – 一个小型的类,会在文件数据上传时处理它。上传处理器在FILE_UPLOAD_HANDLERS
中定义,默认为:
("django.core.files.uploadhandler.MemoryFileUploadHandler","django.core.files.uploadhandler.TemporaryFileUploadHandler",)
MemoryFileUploadHandler
和TemporaryFileUploadHandler
一起提供了Django的默认文件上传行为,将小文件读取到内存中,大文件放置在磁盘中。
你可以编写自定义的处理器,来定制Django如何处理文件。例如,你可以使用自定义处理器来限制用户级别的配额,在运行中压缩数据,渲染进度条,甚至是向另一个储存位置直接发送数据,而不把它存到本地。关于如何自定义或者完全替换处理器的行为,详见编写自定义的上传处理器。
上传数据在哪里储存
在你保存上传文件之前,数据需要储存在某个地方。
通常,如果上传文件小于2.5MB,Django会把整个内容存到内存。这意味着,文件的保存仅仅涉及到从内存读取和写到磁盘,所以非常快。
但是,如果上传的文件很大,Django会把它写入一个临时文件,储存在你系统的临时目录中。在类Unix的平台下,你可以认为Django生成了一个文件,名称类似于/tmp/tmpzfp6I6.upload
。如果上传的文件足够大,你可以观察到文件大小的增长,由于Django向磁盘写入数据。
这些特定值 – 2.5 MB,/tmp
,以及其它 – 都仅仅是”合理的默认值”,它们可以自定义,这会在下一节中描述。
更改上传处理器的行为
Django的文件上传处理器的行为由一些设置控制。详见文件上传设置。
在运行中更改上传处理器
有时候一些特定的视图需要不同的上传处理器。在这种情况下,你可以通过修改request.upload_handlers
,为每个请求覆盖上传处理器。通常,这个列表会包含FILE_UPLOAD_HANDLERS
提供的上传处理器,但是你可以把它修改为其它列表。
例如,假设你编写了ProgressBarUploadHandler
,它会在上传过程中向某类AJAX控件提供反馈。你可以像这样将它添加到你的上传处理器中:
request.upload_handlers.insert(0, ProgressBarUploadHandler())
在这中情况下你可能想要使用list.insert()
(而不是append()
),因为进度条处理器需要在任何其他处理器 之前执行。要记住,多个上传处理器是按顺序执行的。
如果你想要完全替换上传处理器,你可以赋值一个新的列表:
request.upload_handlers = [ProgressBarUploadHandler()]
注意
你只可以在访问
request.POST
或者request.FILES
之前修改上传处理器– 在上传处理工作执行之后再修改上传处理就毫无意义了。如果你在读取request.FILES
之后尝试修改request.upload_handlers
,Django会抛出异常。所以,你应该在你的视图中尽早修改上传处理器。
CsrfViewMiddleware
也会访问request.POST
,它是默认开启的。意思是你需要在你的视图中使用csrf_exempt()
,来允许你修改上传处理器。接下来在真正处理请求的函数中,需要使用csrf_protect()
。注意这意味着处理器可能会在CSRF验证完成之前开始接收上传文件。例如:
from django.views.decorators.csrf import csrf_exempt, csrf_protect@csrf_exempt
def upload_file_view(request):request.upload_handlers.insert(0, ProgressBarUploadHandler())return _upload_file_view(request)@csrf_protect
def _upload_file_view(request):... # Process request
译者:Django 文档协作翻译小组,原文:Overview。
本文以 CC BY-NC-SA 3.0 协议发布,转载请保留作者署名和文章出处。
Django 文档协作翻译小组人手紧缺,有兴趣的朋友可以加入我们,完全公益性质。交流群:467338606。
django 1.8 官方文档翻译: 3-3-1 文件上传相关推荐
- Python+django网页设计入门(6):文件上传与数据导入
前导课程: Python+django网页设计入门(5):自定义用户注册与登录功能 Python+django网页设计入门(4):用户登录与登录验证 Python+django网页设计入门(3):使用 ...
- django 1.8 官方文档翻译: 3-3-4 管理文件
管理文件 这篇文档描述了Django为那些用户上传文件准备的文件访问API.底层的API足够通用,你可以使用为其它目的来使用它们.如果你想要处理静态文件(JS,CSS,以及其他),参见管理静态文件(C ...
- django 1.8 官方文档翻译:6-3 Django异常
Django异常 DJango会抛出一些它自己的异常,以及Python的标准异常. Django核心异常 Django核心异常类定义在django.core.exceptions中. ObjectDo ...
- django 1.8 官方文档翻译:5-1-4 内建的Widget
Widgets Widget 是Django 对HTML 输入元素的表示.Widget 负责渲染HTML和提取GET/POST 字典中的数据. 小贴士 不要将Widget 与表单字段搞混淆.表单字段负 ...
- django 1.8 官方文档翻译: 5-1-1 使用表单
使用表单 关于这页文档 这页文档简单介绍Web 表单的基本概念和它们在Django 中是如何处理的.关于表单API 某方面的细节,请参见表单 API.表单的字段和表单和字段的检验. 除非你计划构建的网 ...
- django文件上传
Django在处理文件上传时,文件数据被打包封装在request.FILES中. 一.简单上传 首先,写一个form模型,它必须包含一个FileField: # forms.py from djang ...
- django 快速实现文件上传
对于web开来说,用户登陆.注册.文件上传等是最基础的功能,针对不同的web框架,相关的文章非常多,但搜索之后发现大多都不具有完整性,对于想学习web开发的新手来说就没办法一步一步的操作练习:对于we ...
- Django实现任意文件上传(最简单的方法)
利用Django实现文件上传并且保存到指定路径下,其实并不困难,完全不需要用到django的forms,也不需要django的models,就可以实现,下面开始实现. 第一步:在模板文件中,创建一个f ...
- django 上传文件夹_django文件上传
1.model中定义属性类型为models.ImageField类型 pic=models.ImageField(upload_to='images/upload/') 2.如果属性类型为ImageF ...
最新文章
- 2021年大数据Spark(二十):Spark Core外部数据源引入
- Java异常处理-----java异常体系
- Python 爬取了马蜂窝的出行数据,告诉你这个夏天哪里最值得去!
- 计算机网络网络层重要概念
- 《现代操作系统》读书笔记
- 计算机一级题库百度云0,全国计算机等级考试一级题库完整.pdf
- 大漠插件最新版7.2111
- 最新行政区划代码省市区数据库-行政区域查询API
- python计算方位角_python如何计算方位角 python计算方位角代码实例
- 【安装windows10 RTX3090 tensorflow的开发环境】
- Shiro框架学习笔记、整合Springboot、redis缓存
- 两个DIV并列排在一个大的DIV中
- Scrum之团队绩效评估
- SAP重置公司代码资产会计(FI-AA)数据-OABL
- 盘点2022初级Java笔试题,选择题,简答题(右滑查看答案)
- 深入理解机器学习——偏差(Bias)与方差(Variance)
- 松下FPXH自动螺丝机程序 昆仑通态触摸屏控触摸,松 下FPXH数据表定位模式,写法新颖
- 数据中心甲方项目管理杂谈
- 【金融项目】尚融宝项目(三)
- 什么是千年虫?计算机如何开始处理日期?都有哪些时间日期格式化?
热门文章
- python3有什么用_Python 3.9的到来到底是意味着什么
- 1018.eclipse工具使用记录
- 【Python】表格文件处理
- 【C语言】删除元素(函数,数组的扫描)
- mysql 利用触发器(Trigger)让代码更简单
- php 遍历目录下的子目录文件,PHP获取目录及子目录下指定后缀的所有文件
- java iso-8859-1_如何在Java中的ISO-8859-1和UTF-8之间转换?
- c语言 tcl.exe 自动登录,Tcl命令操作实验-----(3)---字符串
- 让接口性能轻松翻倍之10条经验
- url中传递url参数|url中特殊字符、?、=无法解析问题