11

我需要压缩任何上载的图像小于500kb的文件大小,我已经在Google上进行了搜索,而我所能看到的是:

 >>> foo = foo.resize((160,300),Image.ANTIALIAS)
 >>> foo.save("path\\to\\save\\image_scaled.jpg",quality=95)

如果我采用这种方法,我将不得不检查压缩后图像是否小于 500kb,如果不是,则选择较低的质量和大小。

有更好的方法吗?

4

2 回答 2

11

JPEG 压缩是不可预测的。你描述的方法,压缩&测量&重试,是我知道的唯一方法。

您可以尝试使用不同质量设置压缩一些典型图像,以了解最佳起点,以及猜测设置更改将如何影响大小的方法。这将使您在没有太多迭代的情况下将最佳尺寸归零。

您还可以将类似文件的对象传递给save不需要写入磁盘的函数,只需计算字节数。确定最佳设置后,您可以再次将其保存到实际文件中。

编辑:这是一个合适的字节计数文件对象的实现。保存后检查size即可。

class file_counter(object):
    def __init__(self):
        self.position = self.size = 0

    def seek(self, offset, whence=0):
        if whence == 1:
            offset += self.position
        elif whence == 2:
            offset += self.size
        self.position = min(offset, self.size)

    def tell(self):
        return self.position

    def write(self, string):
        self.position += len(string)
        self.size = max(self.size, self.position)

编辑 2:这是使用上述方法进行的二进制搜索,以quality在最少的尝试次数中获得最佳结果。

def smaller_than(im, size, guess=70, subsampling=1, low=1, high=100):
    while low < high:
        counter = file_counter()
        im.save(counter, format='JPEG', subsampling=subsampling, quality=guess)
        if counter.size < size:
            low = guess
        else:
            high = guess - 1
        guess = (low + high + 1) // 2
    return low
于 2012-11-15T23:08:59.520 回答
3

我想我会在这里提供我的代码,这样可能会对遇到同样问题的人有所帮助

class PhotoField(forms.FileField, object):

    def __init__(self, *args, **kwargs):
        super(PhotoField, self).__init__(*args, **kwargs)
        self.help_text = "Images over 500kb will be resized to keep under 500kb limit, which may result in some loss of quality"

    def validate(self,image):
        if not str(image).split('.')[-1].lower() in ["jpg","jpeg","png","gif"]:
            raise ValidationError("File format not supported, please try again and upload a JPG/PNG/GIF file")

    def to_python(self, image):
        limit = 500000
        img = Image.open(image.file)
        width, height = img.size
        ratio = float(width) / float(height)
        quality = 100
        while len(image.file.read()) > limit:
            width -= 100
            quality -= 10
            height = int(width / ratio)
            img.resize((width, height), Image.ANTIALIAS)
            img.save(image.file.name, "JPEG", quality=quality)
            image.file = open(image.file.name)
            # reset the file pointer to the beginning so the while loop can read properly
            image.file.seek(0)
        return image

http://james.lin.net.nz/2012/11/19/django-snippet-reduce-image-size-during-upload/

于 2012-11-19T23:49:23.487 回答