版权声明:本文源自简书tianma,转载请务必注明出处: http://www.jianshu.com/p/34e6e3bd233d
最近使用Django1.8.5搭建了一个Web项目,用来生成Android客户端的皮肤apk,相当于一个在线的皮肤apk生成工具,于是就理所当然的需要进行在线的下载apk的操作。由于Android项目比较大,一种主题皮肤对应的apk不止一个,一次性下载多个文件的话,于是选择打包下载。
本文在使用Python在内存中生成zip文件的基础上也进行了小的修改。主要是原文的生产环境是Python2.x,而我使用的是Python3.4,在语法上有些小的变动。
# !user/bin/env python3
# -*-coding : utf-8 -*-
import zipfile
from io import BytesIO
import os
u'''
Create zip file in memory.
'''
class InMemoryZIP(object):
def __init__(self):
# create the in-memory file-like object
self.in_memory_zip = BytesIO()
def append(self, filename_in_zip, file_contents):
""" Appends a file with name filename_in_zip \
and contents of file_contents to the in-memory zip.
"""
# create a handle to the in-memory zip in append mode
zf = zipfile.ZipFile(self.in_memory_zip, 'a',
zipfile.ZIP_DEFLATED, False)
# write the file to the in-memory zip
zf.writestr(filename_in_zip, file_contents)
# mark the files as having been created on Windows
# so that Unix permissions are not inferred as 0000
for zfile in zf.filelist:
zfile.create_system = 0
return self
def appendfile(self, file_path, file_name=None):
""" Read a file with path file_path \
and append to in-memory zip with name file_name.
"""
if file_name is None:
file_name = os.path.split(file_path)[1]
f = open(file_path, 'rb')
file_contents = f.read()
self.append(file_name, file_contents)
f.close()
return self
def read(self):
""" Returns a string with the contents of the in-memory zip.
"""
self.in_memory_zip.seek(0)
return self.in_memory_zip.read()
def writetofile(self, filename):
"""
Write the in-memory zip to a file
"""
f = open(filename, 'wb')
f.write(self.read())
f.close()
if __name__ == '__main__':
imz = InMemoryZIP()
imz.appendfile('a.txt').append('test.txt', 'This is content in test.txt')
imz.writetofile('test.zip')
与原链接中的博文相比主要改动如下:
将原文的import StringIO改成了from io import BytesIO,主要就是Python2.x和Python3.x的格式区别。其实也可以将import StringIO改成from io import BytesIO as StringIO,这样的话在代码中就不需要进行替换,但是可能会误导其他读这段代码的人。
在Django中简单的使用方法为:
#...
def downloadFiles(request):
fns = ["/apk/foo1.apk", "/apk/foo2.apk", "/apk/foo3.apk",]
imz = InMemoryZip()
for fn in fns:
imz.appendfile(fn)
data = img.read()
response = HttpResponse(content_type="application/octet-stream")
response["Content-Disposition"] = "attachment; %s" % "foo.zip"
response["Content-Length"] = len(data)
response.write(data)
return response
与原链接的博文相比改动就是HttpResponse中的参数要使用content_type而不是mimetype
最后引用原文的一段话:
这个方法虽然很方便,不过很耗资源,我试着用它在 Django 里压缩一个 1.4G 的文本文件,差不多用了 8 分钟,期间 CPU 使用率一直是 100%,所以,如果要压缩的是大文件,或者压缩任务比较频繁,可能需要认真处理一下性能问题。
也就是说这个方法适合下载小文件,要是下载的文件较大的话,建议阅读以下文章: