本实验中添加水印的概念其实可以理解为将一张图片中的某个物体或者图案提取出来,然后叠加到另一张图片上。具体的操作思想是通过将原始图片转换成灰度图,并进行二值化处理,去除背景部分,得到一个类似掩膜的图像。然后将这个二值化图像与另一张图片中要添加水印的区域进行“与”运算,使得目标物体的形状出现在要添加水印的区域。最后,将得到的目标物体图像与要添加水印的区域进行相加,就完成了添加水印的操作。这样可以实现将一个图像中的某个物体或图案叠加到另一个图像上,从而实现添加水印的效果。就本实验而言,会用到两个新的组件,一个是模板输入,一个是图像融合。
1. 模板输入
其实,该组件起到的就是图片输入的功能,只不过使用模板输入所输入的图片其实是作为要添加的水印,有了水印的彩色图之后,我们需要用它来制作一个掩模,这就用到了灰度化和二值化,即先灰度化后二值化,这就得到了带有水印图案的掩模。
2. 与运算
有了模板的掩膜之后(也就是二值化图),我们就可以在要添加水印的图像中,根据掩膜的大小切割出一个ROI区域,也就是我们要添加水印的区域,之后让其与模板的掩膜进行与运算,我们知道,与运算的过程中,只要有黑色像素,那么得到的结果图中的对应位置也会是黑色像素。由于模板的掩膜中目标物体的像素值为黑色,所以经过与运算后,就会在ROI区域中得到模板图的形状。
3. 图像融合
将模板的形状添加到水印区域之后,就可以将该图像与原始的模板图进行图像融合。该组件的目的就是将图像对应的数组中的对应元素进行相加(一定要注意这里的两个数组是规格相同的,也就是说要么都是灰度图,要么都是彩图),其过程如下图所示。
因此就可以让原始的模板图与添加模板图形状的ROI区域进行图像融合,得到添加水印的ROI区域。
cv2.bitwise_and(src1, src2, mask=None, dst=None)
功能:用于对两个数组(通常是图像)进行按位与(bitwise AND)操作的函数。
参数:
src1: 第一个输入数组(图像),它可以是单通道或多通道的。
src2: 第二个输入数组(图像),它必须与 src1 有相同的尺寸和类型,或者是可以与 src1 进行广播的数组。
mask: 操作掩码,这是一个可选参数。如果提供,它必须是一个单通道的二进制数组(即只包含 0 和 255 的值)。在运算过程中,只有当掩码对应位置的值为 255 时,src1 和 src2 对应位置的像素才会进行按位与运算;否则,结果数组在该位置的像素值将被设置为 0。
dst: 输出数组,这是一个可选参数。如果提供,运算结果将存储在这个数组中;如果未提供,函数将创建一个新的数组来存储结果。
dst、mask 和输入数组 src1、src2 可以有相同的大小和类型,或者 dst 和 mask 可以是原数组大小的一部分(这种情况下会进行广播)。
cv2.add(src1, src2, dst=None, mask=None, dtype=None)
功能:用于对两个数组(图像)进行逐元素相加操作的函数
参数:
src1: 第一个输入数组(图像)。
src2: 第二个输入数组(图像),它必须与 src1 有相同的尺寸和类型,或者可以与 src1 进行广播。
dst: 输出数组,这是一个可选参数。如果提供,运算结果将存储在这个数组中;如果未提供,函数将创建一个新的数组来存储结果。
mask: 操作掩码,这是一个可选参数。如果提供,它必须是一个单通道的二进制数组(只包含0和255的值,或者布尔数组)。在运算过程中,只有当掩码对应位置的值为255(或True)时,src1和src2对应位置的像素才会进行相加运算;否则,结果数组在该位置的像素值将保持不变(或者根据dtype参数可能被设置为其他值)。
dtype: 输出数组的可选深度(数据类型),当两个输入数组具有相同深度时,这个值将被忽略。它可以是-1,此时输出数组和输入数组有相同的深度,或者它可以是cv2.CV_8U、cv2.CV_16U、cv2.CV_32F等,用于指定输出数组的数据类型。
效果图
import cv2
img = cv2.imread('../1iamge/test2.png')
logo = cv2.imread('../1iamge/test.jpg')logo_gray = cv2.cvtColor(logo,cv2.COLOR_BGR2GRAY)
ret,logo_binary = cv2.threshold(logo_gray,0,255,cv2.THRESH_BINARY_INV)
ROI = img[img.shape[0]-logo.shape[0]:img.shape[0],img.shape[1]-logo.shape[1]:img.shape[1]]
ROI_logo = cv2.bitwise_and(ROI,ROI,mask = logo_binary)
img_logo = cv2.add(ROI_logo,logo)
img[img.shape[0]-logo.shape[0]:img.shape[0],img.shape[1]-logo.shape[1]:img.shape[1]] = img_logo
cv2.imshow("img",img)
cv2.waitKey(0)