django文件的下载

2017-11-05  本文已影响0人  赖三石

django提供文件下载时,若果文件较小,解决办法是先将要传送的内容全生成在内存中,然后再一次性传入Response对象中:

def simple_file_download(request):
    # do something...
    content = open("simplefile", "rb").read()
    return HttpResponse(content)



如果文件非常大时,最简单的办法就是使用静态文件服务器,比如Apache或者Nginx服务器来处理下载。不过有时候,我们需要对用户的权限做一下限定,或者不想向用户暴露文件的真实地址,或者这个大内容是临时生成的(比如临时将多个文件合并而成的),这时就不能使用静态文件服务器了。

templates/index.html

<div class="row">  
      <div class="col-md-8 col-md-offset-2">  
          <br>  
          <P>第一种方法,直接把链接地址指向要下载的静态文件,在页面中点击该链接,可以直接打开该文件,在链接上点击右键,选择“另存为”可以保存该文件到本地硬盘。  
             此方法只能实现静态文件的下载,不能实现动态文件的下载。</P>  
          <a href="{% url 'media' 'uploads/11.png' %}">11.png</a>  
          <br>  
          <br>  
          <p>第二种方法,将链接指向相应的view函数,在view函数中实现下载功能,可以实现静态和动态文件的下载。</p>  
          <a href="{% url 'course:download_file' %}">11.png</a>  
          <br>  
          <br>  
          <br>  
          <p>第三种方法,与第二种方法类似,利用按钮的响应函数实现文件下载功能。</p>  
          <label> 11.png</label><button onclick="window.location.href='{% url 'course:download_file' %}'">Download</button>  
      </div>  
  </div>  

views.py

def download_file(request):  
    # do something  
  
    the_file_name='xxx.png'             #显示在弹出对话框中的默认的下载文件名      
    filename='media/uploads/xxx.png'    #要下载的文件路径  
    response=StreamingHttpResponse(readFile(filename))  
    response['Content-Type']='application/octet-stream'  
    response['Content-Disposition']='attachment;filename="{0}"'.format(the_file_name)  
    return response  
  
def readFile(filename,chunk_size=512):  
    with open(filename,'rb') as f:  
        while True:  
            c=f.read(chunk_size)  
            if c:  
                yield c  
            else:  
                break  



python也提供一个文件包装器,将类文件对象包装成一个迭代器:(未验证)

class FileWrapper:
    """Wrapper to convert file-like objects to iterables"""
    def __init__(self, filelike, blksize=8192):
        self.filelike = filelike
        self.blksize = blksize
        if hasattr(filelike,'close'):
            self.close = filelike.close
    def __getitem__(self,key):
        data = self.filelike.read(self.blksize)
        if data:
            return data
        raise IndexError
    def __iter__(self):
        return self
    def next(self):
        data = self.filelike.read(self.blksize)
        if data:
            return data
        raise StopIteration

使用时views.py的写法

from django.core.servers.basehttp import FileWrapper
from django.http import HttpResponse
import os
def file_download(request,filename):
    wrapper = FileWrapper(file('filepath'))
    response = HttpResponse(wrapper, content_type='application/octet-stream')
    response['Content-Length'] = os.path.getsize(path) ??????
    response['Content-Disposition'] = 'attachment; filename=%s' % filename
    return response
上一篇下一篇

猜你喜欢

热点阅读