首先申明该文使用的django 版本为svn 版本,我每天都在更新 ,python 是2.5,pil 也是最新的版本。
通常情况下用django上传图片我们会定义一个django.forms.Form的子类,比如:
# coding: utf-8
import django.forms as forms
class PictureForm(forms.Form):
# ......
# 图片
imagefile = forms.ImageField()
# ......
然后我们在服务端会定义一个函数来处理图片上传,我们先来看看一般的做法:
# coding: utf-8
from django.http import HttpResponse
def addPicture(request):
if request.method == 'POST':
form = PictureForm(request.POST, request.FILES)
if form.is_valid():
f = request.FILES["imagefile"]
# des_origin_path 为你在服务器上保存原始图片的文件物理路径
des_origin_f = open(des_origin_path, "ab")
for chunk in f.chunks():
des_origin_f.write(chunk)
des_origin_f.close()
# 图片文件被保存在硬盘上后,你可以用pil的Image.open打开进行处理,比如改变大小,加水印等等
在上面这种传统的做法中,是把图片文件保存到硬盘上后,再由pil图形库从硬盘上加载,再进行图片处理,明显多了很多的硬盘IO操作,如果你是一个图片处理量很大的应用,显然不想看到这样的局面。那我们来看看另外一个做法:
# coding: utf-8
from django.http import HttpResponse
import ImageFile
def addPicture(request):
if request.method == 'POST':
form = PictureForm(request.POST, request.FILES)
if form.is_valid():
f = request.FILES["imagefile"]
parser = ImageFile.Parser()
for chunk in f.chunks():
parser.feed(chunk)
img = parser.close()
# 在img被保存之前,可以进行图片的各种操作,在各种操作完成后,在进行一次写操作
img.save("yoursavepath")
在上面的做法中,我们利用pil的ImageFile直接从django接收的数据流中构建出图形对象,进行处理后,在保存,减少了硬盘的读操作。现在网上有些例子使用StringIO对象来作为pil Image.open的输入,经过我多次尝试,在ubuntu以及winxp、centos上都要报异常,我想可能跟pil的版本有关,或许以前的低版本pil的确可以处理。