OpenCV轻松入门:面向Python
上QQ阅读APP看本书,新人免费读10天
设备和账号都新为新人

3.3 按位逻辑运算

逻辑运算是一种非常重要的运算方式,图像处理过程中经常要按照位进行逻辑运算,本节介绍OpenCV中的按位逻辑运算,简称位运算。

在OpenCV内,常见的位运算函数如表3-1所示。

表3-1 常见的位运算函数

3.3.1 按位与运算

在与运算中,当参与与运算的两个逻辑值都是真时,结果才为真。其逻辑关系可以类比图3-4所示的串联电路,只有当两个开关都闭合时,灯才会亮。

图3-4 与运算类比电路

表3-2对与运算算子的不同情况进行了说明,表中使用“and”表示与运算。

表3-2 与运算

按位与运算是指将数值转换为二进制值后,在对应的位置上进行与运算。例如,表3-3展示了两个数值进行按位与运算的示例。

表3-3 按位与运算

在OpenCV中,可以使用cv2.bitwise_and()函数来实现按位与运算,其语法格式为:

        dst = cv2.bitwise_and( src1, src2[, mask]] )

式中:

● dst表示与输入值具有同样大小的array输出值。

● src1表示第一个array或scalar类型的输入值。

● src2表示第二个array或scalar类型的输入值。

● mask表示可选操作掩码,8位单通道array。

按位与操作有如下特点:

● 将任何数值N与数值0进行按位与操作,都会得到数值0。

● 将任何数值N(这里仅考虑8位值)与数值255(8位二进制数是11111111)进行按位与操作,都会得到数值N本身。

可以通过表3-4观察数值N(表中是219)与特殊值0和255进行按位与运算的结果。

表3-4 与特殊值0和255进行按位与运算

根据上述特点,可以构造一幅掩模图像M,掩模图像M中只有两种值:一种是数值0,另外一种是数值255。将该掩模图像M与一幅灰度图像G进行按位与操作,在得到的结果图像R中:

● 与掩模图像M中的数值255对应位置上的值,来源于灰度图像G。

● 与掩模图像M中的数值0对应位置上的值为零(黑色)。

第13章将从另外一个角度对上述情况进行说明,以帮助大家更好地理解掩模及处理方式。

例3.7】使用数组演示与掩模图像的按位与运算。

根据题目要求,编写代码如下:

        import cv2
        import numpy  as np
        a=np.random.randint(0,255, (5,5), dtype=np.uint8)
        b=np.zeros((5,5), dtype=np.uint8)
        b[0:3,0:3]=255
        b[4,4]=255
        c=cv2.bitwise_and(a, b)
        print("a=\n", a)
        print("b=\n", b)
        print("c=\n", c)

运行上述程序,输出结果如下:

        a=
         [[ 39177191  66179]
         [109  28   7  17118]
         [241191  32  72202]
         [229  62   9  65187]
         [103  65207  45121]]
        b=
         [[255255255   0   0]
         [255255255   0   0]
         [255255255   0   0]
         [  0   0   0   0   0]
         [  0   0   0   0255]]
        c=
         [[ 39177191   0   0]
         [109  28   7   0   0]
         [241191  32   0   0]
         [  0   0   0   0   0]
         [  0   0   0   0121]]

从程序可以看出,数组c来源于数组a与数组b的按位与操作。运算结果显示,对于数组c内的值,与数组b中数值255对应位置上的值来源于数组a;与数组b中数值0对应位置上的值为0。

例3.8】构造一个掩模图像,使用按位与运算保留图像中被掩模指定的部分。

在本例中,我们构造一个掩模图像,保留图像lena的头部。

根据题目要求,编写代码如下:

        import cv2
        import numpy  as np
        a=cv2.imread("lena.bmp",0)
        b=np.zeros(a.shape, dtype=np.uint8)
        b[100:400,200:400]=255
        b[100:500,100:200]=255
        c=cv2.bitwise_and(a, b)
        cv2.imshow("a", a)
        cv2.imshow("b", b)
        cv2.imshow("c", c)
        cv2.waitKey()
        cv2.destroyAllWindows()

运行上述程序,输出结果如图3-5所示,左图是原始图像lena,中间的图是掩模图像,右图是按位与结果图像,可以看到,被掩模指定的头部图像被保留在了运算结果中。

图3-5 【例3.8】程序的运行结果

除了需要对灰度图像进行掩模处理,还经常需要针对BGR模式的彩色图像使用掩模提取指定部分。由于按位与操作要求参与运算的数据有相同的通道,所以无法直接将彩色图像与单通道的掩模图像进行按位与操作。一般情况下,可以通过将掩模图像转换为BGR模式的彩色图像,让彩色图像与掩模图像进行按位与操作,实现掩模运算。

例3.9】构造一个掩模图像,使用按位与操作保留图像内被掩模所指定的部分。

根据题目要求,编写代码如下:

        import cv2
        import numpy  as np
        a=cv2.imread("lena.bmp",1)
        b=np.zeros(a.shape, dtype=np.uint8)
        b[100:400,200:400]=255
        b[100:500,100:200]=255
        c=cv2.bitwise_and(a, b)
        print("a.shape=", a.shape)
        print("b.shape=", b.shape)
        cv2.imshow("a", a)
        cv2.imshow("b", b)
        cv2.imshow("c", c)
        cv2.waitKey()
        cv2.destroyAllWindows()

运行上述程序,输出结果如图3-6所示,其中左图是原始图像,中间的图是掩模图像,右图是原始图像和掩模图像按位与后提取的图像。由于本书为黑白印刷,所以为了更好地观察运行效果,请大家亲自上机运行程序。

图3-6 【例3.9】程序的运行结果

同时,程序还会显示如下结果:

        a.shape= (512, 512, 3)
        b.shape= (512, 512, 3)

3.3.2 按位或运算

或运算的规则是,当参与或运算的两个逻辑值中有一个为真时,结果就为真。其逻辑关系可以类比为如图3-7所示的并联电路,两个开关中只要有任意一个闭合时,灯就会亮。

图3-7 或运算类比电路图

表3-5对参与或运算的算子的不同情况进行了说明,表中使用“or”表示或运算。

表3-5 或运算

按位或运算是指将数值转换为二进制值后,在对应的位置上进行或运算。例如,表3-6展示了两个数值进行按位或运算的示例。

表3-6 按位或运算

在OpenCV中,可以使用cv2.bitwise_or()函数来实现按位或运算,其语法格式为:

        dst = cv2.bitwise_or( src1, src2[, mask]] )

式中:

● dst表示与输入值具有同样大小的array输出值。

● src1表示第一个array或scalar类型的输入值。

● src2表示第二个array或scalar类型的输入值。

● mask表示可选操作掩码,8位单通道array值。

3.3.3 按位非运算

非运算是取反操作,满足如下逻辑:

● 当运算数为真时,结果为假。

● 当运算数为假时,结果为真。

表3-7对参与运算算子的不同情况进行了说明,表中使用“not”表示非运算。

表3-7 非运算

按位非运算是指将数值转换为二进制值后,在对应的位置上进行非运算。例如,表3-8展示了按位非运算的示例。

表3-8 按位非运算

在OpenCV中,可以使用函数cv2.bitwise_not()来实现按位取反操作,其语法格式为:

        dst = cv2.bitwise_not( src[, mask]] )

式中:

● dst表示与输入值具有同样大小的array输出值。

● src表示array类型的输入值。

● mask表示可选操作掩码,8位单通道array值。

3.3.4 按位异或运算

异或运算也叫半加运算,其运算法则与不带进位的二进制加法类似,其英文为“exclusive OR”,因此其函数通常表示为xor。

表3-9对参与异或运算的算子的不同情况进行了说明,其中“xor”表示异或运算。

表3-9 异或运算

按位异或运算是指将数值转换为二进制值后,在对应的位置上进行异或运算。例如,表3-10展示了两个数值进行按位异或运算的示例。

表3-10 按位异或运算

在OpenCV中,可以使用函数cv2.bitwise_xor()来实现按位异或运算,其语法格式为:

        dst = cv2.bitwise_xor( src1, src2[, mask]] )

式中:

● dst表示与输入值具有同样大小的array输出值。

● src1表示第一个array或scalar类型的输入值。

● src2表示第二个array或scalar类型的输入值。

● mask表示可选操作掩码,8位单通道array值。