三、OpenCV图像的预处理——二值化与自适应阈值

教程汇总:python基础入门系列

定义: 图像的二值化,就是将图像上的像素点的灰度值设置为0或255,也就是将整个图像呈现出明显的只有黑和白的视觉效果。
一幅图像包括目标物体、背景还有噪声,要想从多值的数字图像中直接提取出目标物体,常用的方法就是设定一个阈值T,用T将图像的数据分成两部分:大于T的像素群和小于T的像素群。这是研究灰度变换的最特殊的方法,称为图像的二值化(Binarization)。

二值化处理是图像预处理中比较常见且重要的一步,通过二值化处理可以凸显需要处理的部分,大大缩减后续图像处理的数据量。

简单阈值(全局阈值)

简单阈值当然是最简单,选取一个全局阈值,然后就把整幅图像分成了非黑即白的二值图像了。函数为Python-OpenCV中提供了阈值(threshold)函数:

cv2.threshold(src, thresh, maxval, type)

  • 第一个参数 src 指原图像,原图像应该是灰度图。

  • 第二个参数 thresh 指用来对像素值进行分类的阈值。

  • 第三个参数 maxval 指当像素值高于(有时是小于)阈值时应该被赋予的新的像素值

  • 第四个参数 type 指不同的阈值方法,这些方法包括:

    • cv2.THRESH_BINARY---------------图(1)
    • cv2.THRESH_BINARY_INV--------图(2)
    • cv2.THRESH_TRUNC---------------图(3)
    • cv2.THRESH_TOZERO-------------图(4)
    • cv2.THRESH_TOZERO_INV-------图(5)

假设下图为被处理的图像素值,破折线为将被阈值化的值;虚线为阈值 。
在这里插入图片描述

经过不同方法阈值化后示意图如下:
在这里插入图片描述
图(1)
大于阈值的像素点的灰度值设定为最大值(如8位灰度值最大为255),灰度值小于阈值的像素点的灰度值设定为0。
在这里插入图片描述
图(2)
大于阈值的像素点的灰度值设定为0,而小于该阈值的设定为255。
在这里插入图片描述
图(3)
像素点的灰度值小于阈值不改变,大于阈值的灰度值的像素点就设定为该阈值。
在这里插入图片描述
图(4)
像素点的灰度值小于该阈值的不进行任何改变,而大于该阈值的部分,其灰度值全部变为0。

在这里插入图片描述
图(5)
像素点的灰度值大于该阈值的不进行任何改变,像素点的灰度值小于该阈值的,其灰度值全部变为0。

使用示例代码如下:

import cv2
import matplotlib.pyplot as plt

# 打开图片
img=cv2.imread('License-plate.jpg')
# 转为灰度图
gray_img = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)

# 二值化
# 参数:1.原图像, 2.进行分类的阈值,3.高于(低于)阈值时赋予的新值,4.方法选择参数
# 返回值:1.得到的阈值,2.阈值化后的图像
retval,threshold_img = cv2.threshold(gray_img,150,255,cv2.THRESH_BINARY)

cv2.imshow("img",img)
cv2.imshow("gray_img",gray_img)
cv2.imshow("threshold_img",threshold_img)
# 对灰度图进行直方图统计显示
plt.hist(gray_img.ravel(), 256, [0, 256])
plt.show()

cv2.waitKey(0)
cv2.destroyAllWindows()

处理后的结果如下:
原图:
在这里插入图片描述
灰度图与二值化图:
在这里插入图片描述
阈值的选择可以通过观察灰度图的直方图来确定,该图像直方图如下:
在这里插入图片描述
可以明显看到是双波峰结构,这种就比较简单了,选取双波峰之间波谷部分的值就行,比如150。

但现实往往是复杂的比如下面这个图就无法使用上面这种简单阈值来处理了
在这里插入图片描述
该图整体偏暗,观察他的直方图可以发现,基本是集中在一起的。
在这里插入图片描述
对于这种图的处理就需要用到更高级的阈值化方法了,下面进行讲解。

自适应阈值(局部阈值):

顾名思义,这个处理用到的阈值是自适应的,而不是整张图共用一个固定值。当同一幅图像上的不同部分的具有不同亮度时。这种情况下我们需要采用自适应阈值。此时的阈值是根据图像上的每一个小区域计算与其对应的阈值。因此在同一幅图像上的不同区域采用的是不同的阈值,从而使我们能在亮度不同的情况下得到更好的结果。
函数:cv2.adaptiveThreshold(src, maxValue, adaptiveMethod, thresholdType, blockSize, C)
参数:

  • 1.原始图像
  • 2.像素值上限
  • 3.自适应方法Adaptive Method:
    • cv2.ADAPTIVE_THRESH_MEAN_C :区域内均值
    • cv2.ADAPTIVE_THRESH_GAUSSIAN_C :区域内像素点加权和,权 重为一个高斯窗口
  • 4.值的赋值方法:只有cv2.THRESH_BINARY 和cv2.THRESH_BINARY_INV
  • 5.Block size:规定区域大小(一个正方形的区域)
  • 6.常数C,与方法有关的参数,阈值等于均值或者加权值减去这个常数(为0相当于阈值 就是求得区域内均值或者加权值)
    • 对方法CV_ADAPTIVE_THRESH_MEAN_C,先求出块中的均值,再减掉C。
    • 对方法 CV_ADAPTIVE_THRESH_GAUSSIAN_C ,先求出块中的加权值(gaussian), 再减掉C。

这种方法理论上得到的效果更好,相当于在动态自适应的调整属于自己像素点的阈值,而不是整幅图像都用一个阈值。
继续以上图为例,示例代码:

import cv2
import matplotlib.pyplot as plt

# 打开图片
img=cv2.imread('bookpage.jpg')
# 转为灰度图
gray_img = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)

# 二值化
# 参数:1.原图像, 2.进行分类的阈值,3.高于(低于)阈值时赋予的新值,4.方法选择参数
# 返回值:1.得到的阈值,2.阈值化后的图像
retval,threshold_img = cv2.threshold(gray_img,20,255,cv2.THRESH_BINARY)
# 自适应阈值
adaptive_threshold_img = cv2.adaptiveThreshold(gray_img, 255, cv2.ADAPTIVE_THRESH_GAUSSIAN_C, cv2.THRESH_BINARY, 35, 3)

cv2.imshow("img",img)
#cv2.imshow("gray_img",gray_img)
cv2.imshow("threshold_img",threshold_img)
cv2.imshow("adaptive_threshold_img",adaptive_threshold_img)
# 对灰度图进行直方图统计显示
#plt.hist(gray_img.ravel(), 256, [0, 256])
#plt.show()

cv2.waitKey(0)
cv2.destroyAllWindows()

简单阈值与自适应阈值的比对:
在这里插入图片描述

扩展:Otsu 二值化

在使用全局阈值时,我们就是随便给了一个数来做阈值,那我们怎么知道我们选取的这个数的好坏呢?答案就是不停的尝试。如果是一副双峰图像(简单来说双峰图像是指图像直方图中存在两个峰)呢?我们岂不是应该在两个峰之间的峰谷选一个值作为阈值?这就是 Otsu 二值化要做的。简单来说就是对一副双峰图像自动根据其直方图计算出一个阈值。(对于非双峰图像,这种方法得到的结果可能会不理想)。
这里用到到的函数还是 cv2.threshold(),但是需要多传入一个参数(flag):cv2.THRESH_OTSU。
这时要把阈值设为 0。然后算法会找到最优阈值,这个最优阈值就是返回值 retVal。如果不使用 Otsu 二值化,返回的retVal 值与设定的阈值相等。
  下面进行代码演示,仍然选取我们第一幅的车牌图像举例。第一种方法,设127 为全局阈值。第二种方法,直接使用 Otsu 二值化。第三种方法,先使用一个 5x5 的高斯核除去噪音,然后再使用 Otsu 二值化。

import cv2

img = cv2.imread('License-plate.jpg')
gray_img = cv2.cvtColor(img,cv2.COLOR_BGR2GRAY)

# 设127 为全局阈值
ret1,th1 = cv2.threshold(gray_img,127,255,cv2.THRESH_BINARY)

# Otsu 滤波
ret2,th2 = cv2.threshold(gray_img,0,255,cv2.THRESH_BINARY+cv2.THRESH_OTSU)
print(ret2)

# 先使用一个 5x5 的高斯核除去噪音,然后再使用 Otsu 二值化
blur = cv2.GaussianBlur(gray_img,(5,5),0)
ret3,th3 = cv2.threshold(blur,0,255,cv2.THRESH_BINARY+cv2.THRESH_OTSU)

cv2.imshow("img",img)
cv2.imshow("th1",th1)
cv2.imshow("th2",th2)
cv2.imshow("th3",th3)

cv2.waitKey(0)
cv2.destroyAllWindows()

得到的打印值ret2:142
结果图如下:
在这里插入图片描述

  • 13
    点赞
  • 74
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
二值化是一种常见的图像处理方法,它将灰度图像转换为黑白图像二值化方法可以根据不同的应用场景进行选择,例如简单阈值二值化自适应阈值二值化等等。在这里,我们介绍一下简单阈值二值化的C++实现及案例。 简单阈值二值化的思路是将灰度图像中灰度值小于一个固定阈值的像素设置为黑色(0),其他像素设置为白色(255)。阈值的选择可以根据实际应用进行调整。 下面是C++代码实现: ```c++ #include <opencv2/opencv.hpp> #include <iostream> using namespace cv; using namespace std; int main(int argc, char** argv) { // 读取图像 Mat src = imread("test.jpg", IMREAD_GRAYSCALE); // 阈值二值化 Mat dst; threshold(src, dst, 100, 255, THRESH_BINARY); // 显示图像 imshow("src", src); imshow("dst", dst); waitKey(0); return 0; } ``` 在上述代码中,使用了OpenCV库进行图像处理。首先,读取了一张灰度图像。然后,使用threshold函数进行阈值二值化处理。函数中的参数含义为:src为输入图像,dst为输出图像,100为阈值,255为超过阈值的像素值,THRESH_BINARY表示二值化处理。 下面是一个简单的案例,将一张彩色图像进行简单阈值二值化处理: ![原图](https://img-blog.csdn.net/20180421210435613?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvemh1amluZ19jb25fYmxvZy5wbmc=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/q/75) ```c++ #include <opencv2/opencv.hpp> #include <iostream> using namespace cv; using namespace std; int main(int argc, char** argv) { // 读取图像 Mat src = imread("test.jpg"); // 灰度化 Mat gray; cvtColor(src, gray, COLOR_BGR2GRAY); // 阈值二值化 Mat dst; threshold(gray, dst, 100, 255, THRESH_BINARY); // 显示图像 imshow("src", src); imshow("dst", dst); waitKey(0); return 0; } ``` 运行结果如下: ![结果](https://img-blog.csdn.net/20180421210501646?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvemh1amluZ19jb25fYmxvZy5wbmc=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/q/75) 从结果可以看出,在简单阈值二值化处理后,图像变成了黑白二值图像

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值