使用Python图像库(PIL)如何将带有alpha通道的图像合成在另一图像上?

17 投票
3 回答
20721 浏览
提问于 2025-04-16 02:05

我有两张图片,都是带有透明通道的。我想把一张图片放在另一张上面,最终得到一张新的图片,这张新图片也要有透明通道,就像它们在图层中叠加一样。我想用Python的图像处理库来实现这个,但如果有其他系统的推荐也很好,甚至一些基本的数学计算方法也很有帮助;我可以用NumPy来处理。

3 个回答

25

Pillow 2.0 现在有一个叫做 alpha_composite 的功能,可以实现这个效果。

img3 = Image.alpha_composite(img1, img2)
31

这段代码看起来可以解决问题:

from PIL import Image
bottom = Image.open("a.png")
top = Image.open("b.png")

r, g, b, a = top.split()
top = Image.merge("RGB", (r, g, b))
mask = Image.merge("L", (a,))
bottom.paste(top, (0, 0), mask)
bottom.save("over.png")
18

我在PIL库里找不到一个叫做alpha composite的函数,所以我尝试用numpy来实现这个功能:

import numpy as np
from PIL import Image

def alpha_composite(src, dst):
    '''
    Return the alpha composite of src and dst.

    Parameters:
    src -- PIL RGBA Image object
    dst -- PIL RGBA Image object

    The algorithm comes from http://en.wikipedia.org/wiki/Alpha_compositing
    '''
    # http://stackoverflow.com/a/3375291/190597
    # http://stackoverflow.com/a/9166671/190597
    src = np.asarray(src)
    dst = np.asarray(dst)
    out = np.empty(src.shape, dtype = 'float')
    alpha = np.index_exp[:, :, 3:]
    rgb = np.index_exp[:, :, :3]
    src_a = src[alpha]/255.0
    dst_a = dst[alpha]/255.0
    out[alpha] = src_a+dst_a*(1-src_a)
    old_setting = np.seterr(invalid = 'ignore')
    out[rgb] = (src[rgb]*src_a + dst[rgb]*dst_a*(1-src_a))/out[alpha]
    np.seterr(**old_setting)    
    out[alpha] *= 255
    np.clip(out,0,255)
    # astype('uint8') maps np.nan (and np.inf) to 0
    out = out.astype('uint8')
    out = Image.fromarray(out, 'RGBA')
    return out

比如说,给定这两张图片,

img1 = Image.new('RGBA', size = (100, 100), color = (255, 0, 0, 255))
draw = ImageDraw.Draw(img1)
draw.rectangle((33, 0, 66, 100), fill = (255, 0, 0, 128))
draw.rectangle((67, 0, 100, 100), fill = (255, 0, 0, 0))
img1.save('/tmp/img1.png')

enter image description here

img2 = Image.new('RGBA', size = (100, 100), color = (0, 255, 0, 255))
draw = ImageDraw.Draw(img2)
draw.rectangle((0, 33, 100, 66), fill = (0, 255, 0, 128))
draw.rectangle((0, 67, 100, 100), fill = (0, 255, 0, 0))
img2.save('/tmp/img2.png')

enter image description here

alpha_composite生成的结果是:

img3 = alpha_composite(img1, img2)
img3.save('/tmp/img3.png')

enter image description here

撰写回答