文件上传

    考虑一个简单的表单,它含有一个FileField

    处理这个表单的视图会在request中接受到上传文件的数据。FILES是个字典,它包含每个FileField的键 (或者 ImageFieldFileField的子类)。这样的话就可以用request.FILES['file']来存放表单中的这些数据了。

    注意request.FILES会仅仅包含数据,如果请求方法为POST,并且发送请求的<form>拥有enctype="multipart/form-data"属性。否则request.FILES为空。

    大多数情况下,你会简单地从request向表单中传递数据,就像绑定上传文件到表单描述的那样。这样类似于:

    1. from django.http import HttpResponseRedirect
    2. from django.shortcuts import render_to_response
    3. from .forms import UploadFileForm
    4. # Imaginary function to handle an uploaded file.
    5. from somewhere import handle_uploaded_file
    6. def upload_file(request):
    7. if request.method == 'POST':
    8. form = UploadFileForm(request.POST, request.FILES)
    9. if form.is_valid():
    10. return HttpResponseRedirect('/success/url/')
    11. else:
    12. form = UploadFileForm()

    注意我们必须向表单的构造器中传递request.FILES。这是文件数据绑定到表单的方法。

    这里是一个普遍的方法,可能你会采用它来处理上传文件:

    1. def handle_uploaded_file(f):
    2. with open('some/file/name.txt', 'wb+') as destination:
    3. for chunk in f.chunks():
    4. destination.write(chunk)

    遍历UploadedFile.chunks(),而不是使用read(),能确保大文件并不会占用系统过多的内存。

    如果你在Model上使用FileField保存文件,使用ModelForm可以让这个操作更加容易。调用form.save()的时候,文件对象会保存在相应的FileFieldupload_to参数指定的地方。

    如果你手动构造一个对象,你可以简单地把文件对象从request.FILE赋值给模型:

    1. from django.http import HttpResponseRedirect
    2. from django.shortcuts import render
    3. from .forms import UploadFileForm
    4. from .models import ModelWithFileField
    5. def upload_file(request):
    6. if request.method == 'POST':
    7. form = UploadFileForm(request.POST, request.FILES)
    8. if form.is_valid():
    9. instance.save()
    10. return HttpResponseRedirect('/success/url/')
    11. else:
    12. form = UploadFileForm()

    上传处理器

    当用户上传一个文件的时候,Django会把文件数据传递给上传处理器 – 一个小型的类,会在文件数据上传时处理它。上传处理器在FILE_UPLOAD_HANDLERS中定义,默认为:

    1. ("django.core.files.uploadhandler.MemoryFileUploadHandler",
    2. "django.core.files.uploadhandler.TemporaryFileUploadHandler",)

    MemoryFileUploadHandlerTemporaryFileUploadHandler一起提供了Django的默认文件上传行为,将小文件读取到内存中,大文件放置在磁盘中。

    你可以编写自定义的处理器,来定制Django如何处理文件。例如,你可以使用自定义处理器来限制用户级别的配额,在运行中压缩数据,渲染进度条,甚至是向另一个储存位置直接发送数据,而不把它存到本地。关于如何自定义或者完全替换处理器的行为,详见编写自定义的上传处理器

    在你保存上传文件之前,数据需要储存在某个地方。

    通常,如果上传文件小于2.5MB,Django会把整个内容存到内存。这意味着,文件的保存仅仅涉及到从内存读取和写到磁盘,所以非常快。

    但是,如果上传的文件很大,Django会把它写入一个临时文件,储存在你系统的临时目录中。在类Unix的平台下,你可以认为Django生成了一个文件,名称类似于/tmp/tmpzfp6I6.upload。如果上传的文件足够大,你可以观察到文件大小的增长,由于Django向磁盘写入数据。

    这些特定值 – 2.5 MB,/tmp,以及其它 — 都仅仅是”合理的默认值”,它们可以自定义,这会在下一节中描述。

    有时候一些特定的视图需要不同的上传处理器。在这种情况下,你可以通过修改request.upload_handlers,为每个请求覆盖上传处理器。通常,这个列表会包含FILE_UPLOAD_HANDLERS提供的上传处理器,但是你可以把它修改为其它列表。

    例如,假设你编写了ProgressBarUploadHandler,它会在上传过程中向某类AJAX控件提供反馈。你可以像这样将它添加到你的上传处理器中:

    在这中情况下你可能想要使用list.insert()(而不是append()),因为进度条处理器需要在任何其他处理器 之前执行。要记住,多个上传处理器是按顺序执行的。

    如果你想要完全替换上传处理器,你可以赋值一个新的列表:

    1. request.upload_handlers = [ProgressBarUploadHandler()]
    1. from django.views.decorators.csrf import csrf_exempt, csrf_protect
    2. @csrf_exempt
    3. def upload_file_view(request):
    4. request.upload_handlers.insert(0, ProgressBarUploadHandler())
    5. return _upload_file_view(request)
    6. @csrf_protect