问答文章1 问答文章501 问答文章1001 问答文章1501 问答文章2001 问答文章2501 问答文章3001 问答文章3501 问答文章4001 问答文章4501 问答文章5001 问答文章5501 问答文章6001 问答文章6501 问答文章7001 问答文章7501 问答文章8001 问答文章8501 问答文章9001 问答文章9501

Django怎么实现文件下载功能

发布网友 发布时间:2023-07-29 09:12

我来回答

1个回答

热心网友 时间:2023-09-18 13:57


这次给大家带来Django怎么实现文件下载功能,Django实现文件下载功能的注意事项有哪些,下面就是实战案例,一起来看一下。
基于Django建立的网站,如果提供文件下载功能,最简单的方式莫过于将静态文件交给Nginx等处理,但有些时候,由于网站本身逻辑,需要通过Django提供下载功能,如页面数据导出功能(下载动态生成的文件)、先检查用户权限再下载文件等。因此,有必要研究一下文件下载功能在Django中的实现。

最简单的文件下载功能的实现

将文件流放入HttpResponse对象即可,如:
def file_download(request):
# do something...
with open('file_name.txt') as f:
c = f.read()
return HttpResponse(c)
这种方式简单粗暴,适合小文件的下载,但如果这个文件非常大,这种方式会占用大量的内存,甚至导致服务器崩溃

更合理的文件下载功能

Django的HttpResponse对象允许将迭代器作为传入参数,将上面代码中的传入参数c换成一个迭代器,便可以将上述下载功能优化为对大小文件均适合;而Django更进一步,推荐使用 StreamingHttpResponse对象取代HttpResponse对象,StreamingHttpResponse对象用于将文件流发送给浏览器,与HttpResponse对象非常相似,对于文件下载功能,使用StreamingHttpResponse对象更合理。

因此,更加合理的文件下载功能,应该先写一个迭代器,用于处理文件,然后将这个迭代器作为参数传递给StreaminghttpResponse对象,如:
from django.http import StreamingHttpResponse
def big_file_download(request):
# do something...

def file_iterator(file_name, chunk_size=512):
with open(file_name) as f:
while True:
c = f.read(chunk_size)
if c:
yield c
else:
break

the_file_name = "file_name.txt"
response = StreamingHttpResponse(file_iterator(the_file_name))

return response
文件下载功能再次优化

上述的代码,已经完成了将服务器上的文件,通过文件流传输到浏览器,但文件流通常会以乱码形式显示到浏览器中,而非下载到硬盘上,因此,还要在做点优化,让文件流写入硬盘。优化很简单,给StreamingHttpResponse对象的Content-Type和Content-Disposition字段赋下面的值即可,如:
response['Content-Type'] = 'application/octet-stream'
response['Content-Disposition'] = 'attachment;filename="test.pdf"'
完整代码如下:
from django.http import StreamingHttpResponse
def big_file_download(request):
# do something...
def file_iterator(file_name, chunk_size=512):
with open(file_name) as f:
while True:
c = f.read(chunk_size)
if c:
yield c
else:
break

the_file_name = "big_file.pdf"
response = StreamingHttpResponse(file_iterator(the_file_name))
response['Content-Type'] = 'application/octet-stream'
response['Content-Disposition'] = 'attachment;filename="{0}"'.format(the_file_name)
return response
具体导出文件格式

导出Excel表格

1. 首先是直接导出Excel表格

首先获取要导出的数据、以列表方式保存。

然后将数据写入到Excel,以流的方式返回到页面下载。
import xlwt
import io
import json
from django.http import HttpResponse
def set_style(name, height, bold=False):
style = xlwt.XFStyle() # 初始化样式
font = xlwt.Font() # 为样式创建字体
font.name = name # 'Times New Roman'
font.bold = bold
font.color_index = 000
font.height = height
style.font = font
# 设置单元格边框
# borders= xlwt.Borders()
# borders.left= 6
# borders.right= 6
# borders.top= 6
# borders.bottom= 6
# style.borders = borders
# 设置单元格背景颜色
# pattern = xlwt.Pattern()
# 设置其模式为实型
# pattern.pattern = pattern.SOLID_PATTERN
# 设置单元格背景颜色
# pattern.pattern_fore_colour = 0x00
# style.pattern = pattern
return style
def write_excel(data, name, header):
# 打开一个Excel工作簿
file = xlwt.Workbook()
# 新建一个sheet,如果对一个单元格重复操作,会引发异常,所以加上参数cell_overwrite_ok=True
table = file.add_sheet(name, cell_overwrite_ok=True)
if data is None:
return file
# 写标题栏
row0 = [u'业务', u'状态', u'北京', u'上海', u'广州', u'深圳', u'状态小计']
for i in range(0, len(row0)):
table.write_merge(0, 0, i, i, row0[i], set_style('Times New Roman', 220, True))
table.write_merge(0, 2, 7, 9, "单元格合并", set_style('Times New Roman', 220, True))
"""
table.write_merge(x, x + m, y, w + n, string, sytle)
x表示行,y表示列,m表示跨行个数,n表示跨列个数,string表示要写入的单元格内容,style表示单元格样式。其中,x,y,w,h,都是以0开始计算的。
"""
l = 0
n = len(header)
# 写入数据
for line in data:
for i in range(n):
table.write(l, i, line[header[i]])
l += 1
# 直接保存文件
# file.save("D:/excel_name.xls")
# 写入IO
res = get_excel_stream(file)
# 设置HttpResponse的类型
response = HttpResponse(content_type='application/vnd.ms-excel')
from urllib import parse
response['Content-Disposition'] = 'attachment;filename=' + parse.quote("excel_name") + '.xls'
# 将文件流写入到response返回
response.write(res)
return response
def get_excel_stream(file):
# StringIO操作的只能是str,如果要操作二进制数据,就需要使用BytesIO。
excel_stream = io.BytesIO()
# 这点很重要,传给save函数的不是保存文件名,而是一个BytesIO流(在内存中读写)
file.save(excel_stream)
# getvalue方法用于获得写入后的byte将结果返回给re
res = excel_stream.getvalue()
excel_stream.close()
return res
2. 导出json文件

导出json文件不像Excel那么麻烦,只需要拼接json格式数据即可,直接导出到本地还是很简单,但是导出到网页,怎么像导出excel一样不保存到本地,直接将流返回?
def write_json(data):
try:
json_stream = get_json_stream(data)
response = HttpResponse(content_type='application/json')
from urllib import parse
response['Content-Disposition'] = 'attachment;filename=' + parse.quote("test") + '.json'
response.write(json_stream)
return response
except Exception as e:
print(e)
def get_json_stream(data):
# 开始这里我用ByteIO流总是出错,但是后来参考廖雪峰网站用StringIO就没问题
file = io.StringIO()
data = json.mps(data)
file.write(data)
res = file.getvalue()
file.close()
return res
3. 导出压缩包

由于导出两个文件无法同时都返回,所以考虑将这两个文件放入包中,然后将包以流的方式返回。

思考?此时导出的是zip包中,我怎么将这两个文件流写入zip中,好像有点不太合理。后来在老大指导下先将要打包的文件保存到本地,打包到zip后,将本地的文件删除,随后将该zip文件流读取,写入到response,返回zip文件流。
def write_zip(e_data, j_data, export_name):
try:
# 保存到本地文件
# 返回文件名,注意此时保存的方法和前面导出保存的json、excel文件区别
j_name = write_json(j_data, export_name[1])
e_name = write_excel(e_data, export_name[1])
# 本地文件写入zip,重命名,然后删除本地临时文件
z_name='export.zip'
z_file = zipfile.ZipFile(z_name, 'w')
z_file.write(j_name)
z_file.write(e_name)
os.remove(j_name)
os.remove(e_name)
z_file.close()
# 再次读取zip文件,将文件流返回,但是此时打开方式要以二进制方式打开
z_file = open(z_name, 'rb')
data = z_file.read()
z_file.close()
os.remove(z_file.name)
response = HttpResponse(data, content_type='application/zip')
from urllib import parse
response['Content-Disposition'] = 'attachment;filename=' + parse.quote(z_name)
return response
except Exception as e:
logging.error(e)
print(e)相信看了本文案例你已经掌握了方法,更多精彩请关注Gxl网其它相关文章!
推荐阅读:
vue使用xe-utils函数库的步奏详解

vue2.0 路由不显示router-view如何处理

Native怎么使用fetch实现图片上传功能
声明声明:本网页内容为用户发布,旨在传播知识,不代表本网认同其观点,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。E-MAIL:11247931@qq.com
国外留学有用吗 花钱出国留学有用吗 !这叫什么号 百万医疗赔付后是否可以续保 前一年理赔过医疗险还能续保吗? 医疗住院险理赔后还能购买吗? 女生多大后可以不在长身高? 如何不用软件把手机投屏到电脑上手机屏幕怎样投放到电脑上 战时拒绝、故意延误军事订货罪既遂的处罚? 战时故意延误军事订货罪处罚标准 复仇风暴建议买哪个颜色 Nginx的配置文件怎样分段下载 洛宁县文昌路在什么地方 ...房租三个月交一次为什么还要提前一个月交?这样是否有道理,我很困... 怎么注销 怎么强制注销 怎么注销掉手机号 兰蔻菁纯粉底液后需用定妆粉吗 刚栽的吊兰多久能生根 TMB300A魔盒和电脑显示屏用HDMI转VGA线连接显示屏为什么黑屏 浦城哪家菜馆的菜好吃 福州哪里有好吃的莆田菜馆 泰山天庭乐园门票价格是多少 米三关于手机里点了安卓版本几次后出现了一下图标,请问这是什么? 传奇私服 IGEM2 SKYM2 区分 安徽省三本大学一览表 梦见抽嘴巴子 杭州市1991年老年人口占比 平时可以经常用姜水洗澡会不会不好 坐月子用姜泡水洗澡可以吗 砖才贴1年左右大部分变颜色,是什么情况 ...一个多月慢慢地扩大,不知道是渗水还是什么原因,问楼下天花板看没发现... 铁路软件工程待遇 笑歌行译文 | 注释 | 赏析 如何分分钟开发高保真交互原型_html/css_WEB-ITnose lunacy的教程 丹阳到浙江舟山的高铁 如何分辨土猪肉 梦见领导要离婚的预兆 昨天装了精简版的系统就这样了能解决吗? 杜甫有哪些千古名句 孩子胆子小如何引导孩子胆子小怎么办 苏宁金融怎么贷款 “一”表示什么符号? 淘宝图片被盗用如何被举报 qq龙王标识怎么关闭 电动机注润滑油咀杆的作用 电机黄油嘴破了影响运行吗 没有加油咀的电动机轴承还需要定期加润滑脂吗? 抗氧化指的是什么?