图像的算数运算是对图像进行加减运算,而图像的逻辑运算是对图像进行与、或、非、异或等逻辑运算。通过算术运算可以让图像来达到图像增强的效果;通过逻辑运算对图像进行分割、图像增强、图像识别、图像复原等操作
图像的加运算可以使用“+”或者OpenCV自带函数cv.add()来实现对图像的加运算。进行算术运算与逻辑运算的图像的分辨率和通道数需要一样。那么“+”和cv.add()有区别吗?答案是肯定的。先上代码看“+”与cv.add()的效果有什么不同:
'''
+对两个图像进行操作时,像素大于255取余数,所以相加后的图形会变暗
add对两个图形进行操作时,像素大于255时保持255,所以相加后的图像会变亮
'''
import cv2 as cv
img = cv.imread('zzf1.jpg')
img = cv.resize(img, (600,800))
cv.imshow('zzf', img)
cv.waitKey(0)
cv.destroyAllWindows()
# 用+对图像相加
img1 = img + img
cv. imshow('+', img1)
# 用add对图像相加
img2 = cv.add(img, img)
cv.imshow('add', img2)
cv.waitKey(0)
cv.destroyAllWindows()
运行结果如下图:
大家可以看的出“+”的加运算让图像变暗了,而cv.add()的加运算让图像变得更亮了。这是为什么呢?主要是当两个图像相加时,灰度值会出现大于255的情况。对于“+”大于255是取余数,比如156+120结果就不是276而是21,;而cv.add()当相加灰度值大于255,默认为255,即白色,所以达到了图像增强变亮的效果。
图像的加运算是当两个不同图像进行相加时,两个图片灰度值所占比重不同,对两个图片进行权值分配。使得想要更突显的部分权值更大,需要淡化的权值略小。主要通过cv.addWeighted()对图像进行操作,函数具体参数介绍见OpenCV-Python计算机视觉函数_一只会飞的猪️的博客-CSDN博客
上代码:
import cv2 as cv
# 读取图片1
img = cv.imread('zzf1.jpg')
img1 = cv.resize(img, (400, 600))
cv.imshow('reba', img1)
# 读取图片2
img2 = cv.imread('zzf2.jpg')
img3 = cv.resize(img2, (400, 600))
cv.imshow('dilireba', img3)
# 对图像加权相加
img4 = cv.addWeighted(img1, 0.6, img3, 0.4, 0)
cv.imshow('addweighted', img4)
cv.waitKey(0)
cv.destroyAllWindows()
运行结果如下图:
可以看得出来,两个图像进行不同权值的相加后产生的效果很明显。一定要注意:通过cv.resize()将两个图像的大小修改成一致,才能进行运算。
当求两张图像差异的时候就会用到图像的减运算,减法运算通过cv.subtract()进行减法操作,函数的具体参数见OpenCV-Python计算机视觉函数_一只会飞的猪️的博客-CSDN博客。老规矩,上代码:
import cv2 as cv
# 读取图片1
img = cv.imread('R-C.jpg')
img1 = cv.resize(img, (600 , 400))
cv.imshow('reba', img1)
# 读取图片2
img2 = cv.imread('reba.jpg')
img3 = cv.resize(img2, (600, 400))
cv.imshow('dilireba', img3)
# 对两图像相减
img4 = cv.subtract(img1, img3)
cv.imshow('subtract', img4)
cv.waitKey(0)
cv.destroyAllWindows()
运行结果如下图:
其实我个人感觉图像的减运算并没有什么用,而且影响美观,减法做出来的图片有点吓人。我思考了一下,应该是选取的图片差异不大导致的相减未能显示出太多的差异。不过能明显看出来第二张图上的花被很好的显示在第三张图片上,验证了确实能显示图片差异的功能。
图像的逻辑运算主要包括逻辑与、逻辑或、逻辑非、逻辑异或等相关逻辑运算。分别通过OpenCV内置函数cv.bitwise_and()、cv.bitwise_or()、cv.bitwise_not()、cv.bitwise_xor()来进行运算,具体用法看我这个专栏的开篇之作函数介绍。一一举例,代码如下:
import cv2 as cv
# 读取图片1
img = cv.imread('didi.jpg')
img1 = cv.resize(img, (600 , 400))
cv.imshow('img1', img1)
# 读取图片2
img2 = cv.imread('baba.jpg')
img3 = cv.resize(img2, (600, 400))
cv.imshow('img2', img3)
# 逻辑与运算
img4 = cv.bitwise_and(img1,img3)
cv.imshow('and', img4)
# 逻辑或运算
img5 = cv.bitwise_or(img1, img3)
cv.imshow('or', img5)
# 逻辑非运算
img5 = cv.bitwise_not(img1)
cv.imshow('not', img5)
# 逻辑异或运算
img6 = cv.bitwise_xor(img1,img3)
cv.imshow('xor', img4)
cv.waitKey(0)
cv.destroyAllWindows()
运行结果如下:
通过对比可以看出各个逻辑变换带来的不同效果。
与运算可以用来截取一张图的某个区域,应用代码如下:
import cv2 as cv
import numpy as np
# 读取图片
img = cv.imread('pangdi.jpg')
cv.imshow('reba', img)
# 生成掩膜图像
mask = np.zeros(img.shape, dtype=np.uint8)
mask[200:400, 400:600] = 255
cv.imshow('mask', mask)
# 进行逻辑与运算,截取重要部分
img1 = cv.bitwise_and(img, mask)
cv.imshow('bitwise_and', img1)
cv.waitKey(0)
cv.destroyAllWindows()
运行结果如下:
黑色的像素值为0不管与什么进行逻辑与结果都是0,即黑色,通过设置白色窗口的位置大小获取目标区域。但必须注意截取图片的mask需要与被截取图像分辨率、通道数一致,
逻辑异或可以用来对图像加密和解密。
加密:图片与密钥异或运算
解密:加密图像与密钥异或运算
代码如下:
import cv2 as cv
import numpy as np
img = cv.imread('zzf3.jpg')
img = cv.resize(img, (800, 500))
cv.imshow('yuantu', img) # 显示原图
key = np.random.randint(0, 256, size=img.shape , dtype=np.uint8)
cv.imshow('key', key) # 显示密钥
# 对图像进行加密
img1 = cv.bitwise_xor(img, key) #显示加密后图形
cv.imshow('encryption', img1)
# 对加密图像解密
img2 = cv.bitwise_xor(img1, key)
cv.imshow('decryption', img2) # 显示解密后图形
cv.waitKey(0)
cv.destroyAllWindows()
运行结果:
大家可以看到,确实实现了图像的加密与解密。各个逻辑运算都有自己的用途,大家感兴趣可以自己搜索了解一下,我这里只是给大家抛砖引玉。
总结:这一篇主要讲了图像的算术运算与逻辑运算的运算方法和一些简单的作用介绍。再强调一下,图像运算的时候一定要保证两个图形的分辨率和通道数一致才能进行运算。一些函数的参数及用法不知道的可以点击OpenCV-Python计算机视觉函数_一只会飞的猪️的博客-CSDN博客去了解一下,这个是 笔者总结的一些OpenCV库,后续还会不断更新。特别感谢俗人嗷的博客_CSDN博客-领域博主提供了珍藏多年的张子枫图片。