1

注意谷歌索引的每张图片都有一个小缩略图。这些缩略图是:

  1. 大小小于 10KB。
  2. 宽度/高度的比例与原始图像中的比例相同。

我想编写一个函数(在 python 中),它将获取图像并使用这些属性创建一个缩略图。

编辑:所以现在有 3 个答案,而且都对了一半。

我需要一个功能,不仅可以按比例调整图像大小,还可以确保文件大小小于 10KB。我怎么做?

4

3 回答 3

3

这篇简短的文章中,Mike Fletcher 和 effbot 展示了(并讨论了详细的变体)一种出色的方法来完成您想要完成的任务。

编辑:至于 10K 的要求,很难(至少可以说;-)根据图像的格式来预测图像的压缩程度,因为今天的压缩算法是如此微妙。如果您希望缩略图尽可能大(以像素为单位),同时满足 <10K 的要求,您可能必须使用“试错法”方法对缩放因子进行连续更精确的猜测,直到达到可接受的结果。

例如,这里有一个“二分搜索”方法来获得正确的大小(可能有更好的策略!),有足够的打印语句 &c 来解释发生了什么......:

import Image
import cStringIO
import math
import os
import stat

# try no more than 10 times, then give up
MAX_TRIES = 10

def getThumbnail(filename, max_bytes=(10*1024)):
    '''Get a thumbnail image of filename, <max_bytes'''
    original_size = os.stat(filename)[stat.ST_SIZE]
    print "Original file size: %.1f KB" % (original_size/1024.)
    image = Image.open(filename)
    image.load()
    print "Original image size: %dx%d pixels" % image.size
    min_bytes = int(0.9 * max_bytes)
    largest_side = max(image.size)
    smallest_side = 16
    for attempt in range(MAX_TRIES):
        try_side = (largest_side + smallest_side) / 2
    print "Attempt #%d of %d" % (attempt+1, MAX_TRIES)
    print "Side must be within [%d:%d], now trying %d" % (
        smallest_side, largest_side, try_side)
    thumb = image.copy()
    thumb.thumbnail((try_side,try_side), Image.ANTIALIAS)
    afile = cStringIO.StringIO()
    thumb.save(afile, "PNG")
    resulting_size = len(afile.getvalue())
    afile.close()
    print "Reduced file size: %.1f KB" % (resulting_size/1024.)
    print "Reduced image size: %dx%d pixels" % thumb.size
    if min_bytes <= resulting_size <= max_bytes:
        print "Success!"
        return thumb
    elif resulting_size > max_bytes:
        print "Too large (>%d), reducing more" % max_bytes
        largest_side = try_side
    else:
        print "Too small (<%d), reducing less" % min_bytes
        smallest_side = try_side
    print "too many attempts, returning what I've got!"
    return thumb

def main():
    thumb = getThumbnail("pyth.png")
    print "Reduced image size: %dx%d pixels" % thumb.size
    print "Saving to thumb.png"
    thumb.save("thumb.png")
    thumb_size = os.stat("thumb.png")[stat.ST_SIZE]
    print "Reduced file size: %.1f KB" % (thumb_size/1024.)
    print "Done, bye!"

if __name__ == '__main__':
    main()
于 2009-07-02T15:32:59.003 回答
1

您是否阅读过PIL 文档?有一个 image.thumbnail 方法。

于 2009-07-02T15:30:37.480 回答
1

使用 PIL,请参阅此处的示例代码以调整大小保持纵横比

如何使用 PIL 调整图像大小并保持其纵横比?

看看如何在目录上做类似的事情

调整目录中的图像大小

因此,上面的链接描述了如何使用 PIL 调整图像大小,现在谈到您的最大大小为 10KB 的问题,这可以轻松实现,例如

假设所需尺寸为 100x100,并且我们使用 JPEG 压缩,100% JPEG 质量每像素大约需要 9 位(请参阅http://en.wikipedia.org/wiki/JPEG),这意味着 100x100 图像的尺寸为 100x100x9/( 1024x8) = 11KB,所以在 quality=100 时你几乎达到了你的目标,但如果你仍然想要 10KB,你可以设置 quality=90,一般你可以将 quality 作为参数传递给 resize 函数并将质量降低 10% 如果图像大小高于 10KB,但我认为这不是必需的,在 90% 的质量下,所有 JPEG 图像都小于 10KB。

另请注意,如果没有压缩,RGB 图像的图像大小也只有 30KB,如果您将大小减小到 60x60 像素,图像大小将只有 10KB,无需任何压缩,即您可以使用 bmp 图像,如果您想要更小尺寸但无损压缩,您可以选择巴布亚新几内亚。

因此,总而言之,您的 10KB 目标很容易实现。

于 2009-07-02T16:11:54.877 回答