写点什么

Python OpenCV 图像处理之直方图相关知识细节,学点细的

发布于: 2021 年 07 月 30 日

Python OpenCV 365 天学习计划,与橡皮擦一起进入图像领域吧。本篇博客是这个系列的第 50 篇。

学在前面

直方图在之前的博客中已经学习过一部分内容了,具体可以自行去回顾。


直方图是图像处理过程中的一个分析工具,是使用灰度值或者从灰度级的角度统计图像的特征。


如果从统计的角度看,直方图统计了图像各个灰度级出现的次数,直方图横坐标是像素灰度级,纵坐标是该灰度级的个数。


如果存在一个 5x5 的灰度图,那直方图就是对其中的各个像素值进行统计。




上面 2 张图就是关于直方图统计的说明,但直方图直观的呈现不是表格,而是图形,所以将上述数据进行绘制,即可得到直方图。直方图中 x 轴表示的是 8 位位图的 256 个灰度级,y 轴表示的是对应灰度级的像素点格式。


基于此还要补充一个知识,叫做归一化的直方图,在很多时候,直方图中 x 轴表示灰度级,但是 y 轴表示灰度级出现的频率,具体就是将次数/总数,例如上表可以修改为:



直方图概念理解之后,还有几个新词需要注意下,DIMSBINSRANGE


  • DIMS:基于目前知识,只有一个灰度值,该值为 1,它表示的是收集直方图时,收集的参数数量;

  • RANGE:统计灰度级范围,对于灰度图,范围是 [0,255]

  • BINS:参数子级数目,大概含义可以理解为,将刚才的灰度级做分组处理,一般保持默认即可。


对于 BINS 参数在细说一下,对于灰度图像,灰度级的区间是 [0,255],其 BINS 默认是 256,如果按照 16 个灰度级为一组,可以分为 16 个 BINS

直方图绘制

Python OpenCV 图像处理之图像直方图,取经之旅第 25 天 本篇博客已经对直方图的绘制进行了说明,先来整体回顾一下。


代码基于 plt.hist() 函数实现直方图,该函数原型为 plt.hist(X,BINS)


运行下述代码,找图片的时候,尽量找颜色分布比较平均的,不要白色或者黑色所占区域特别大,否则得到的直方图效果不明显。


import cv2 as cvimport matplotlib.pyplot as plt
src = cv.imread("./2.png", 0)cv.imshow("src", src)
plt.hist(src.ravel(), 256)plt.show()
cv.waitKey()cv.destroyAllWindows()
复制代码



上述代码还有一个需要注意的就是 src.ravel() 函数了,该函数用于将二维数组降成一维数组,例如下图所示。



如果将 BINS 设置为 16 得到的结果如下。


Python OpenCV 中的直方图绘制

在 OpenCV 中使用 cv.calcHist 函数计算图像的直方图,本文重点说明一下该函数的返回值。


import cv2 as cvimport matplotlib.pyplot as plt
src = cv.imread("./2.png", 0)cv.imshow("src", src)hist = cv.calcHist([src], [0], None, [256], [0, 255])print(hist)print(len(hist))cv.waitKey()cv.destroyAllWindows()
复制代码


hist 输出格式如下:


# 省略一部分输出数据…… [ 115.] [ 122.] [ 115.] [ 142.] [ 148.] [ 152.] [  61.] [   6.] [   0.]]256
复制代码


可以看到最后返回的 hist 是 256 个数字组成的数组,该数组内的元素是各个灰度级的统计个数。得到 hist 之后,就可以通过 plt.plot 函数将其绘制出来啦。


import cv2 as cvimport matplotlib.pyplot as plt
src = cv.imread("./2.png", 0)cv.imshow("src", src)ret_hist = cv.calcHist([src], [0], None, [256], [0, 255])print(ret_hist)print(len(ret_hist))plt.plot(ret_hist)plt.show()cv.waitKey()cv.destroyAllWindows()
复制代码


掩膜绘制直方图

接下来说明一下掩膜绘制直方图,通过掩膜就是选择图像的一部分区域进行绘制,掩膜白色区域表示透明,可显示图片,黑色区域表示不透明,无法显示图片。


掩膜的原理也可以翻阅以前的博客进行学习,总结下来就是下面两句话


  1. 原图像与掩膜中黑色位置对应的部分,这部分就不再显示了,灰度值被置为 0;

  2. 原图像与掩膜中白色位置对应的部分,保留原值。


import cv2 as cvimport matplotlib.pyplot as pltimport numpy as npsrc = cv.imread("./2.png", 0)cv.imshow("src", src)print(src.shape)mask = np.zeros(src.shape, np.uint8)# 前面是行,后面是列mask[10:150, 40: 200] = 255mask_img = cv.bitwise_and(src, mask)cv.imshow("mask", mask_img)ret_hist = cv.calcHist([src], [0], None, [256], [0, 255])mask_hist = cv.calcHist([src], [0], mask, [256], [0, 255])
plt.plot(ret_hist)plt.plot(mask_hist)plt.show()cv.waitKey()cv.destroyAllWindows()
复制代码


直方图均衡化相关知识补充

直方图均衡化,可以使图像拥有全部可能的灰度级,让像素值的灰度均匀分布,实现的具体步骤如下。


  1. 计算累计直方图

  2. 对累计直方图进行转换


可以参考下述灰度矩阵,该图像大小为 5x5,具有 8 个灰度级,范围是 [0,7],计算归一化的统计直方图与累计统计直方图分别如下。



基于上图,可以对原有灰度级空间进行转换,分为两种一种是在原有的灰度空间范围,即 [0,7],第二种是在更大范围进行转换。


首先看一下在原有范围内实现均衡化,用当前的累计概率乘以 7(灰度级的最大值),得到新的灰度级。



新灰度级与原灰度级个数可以对比得出。


未均衡化之前,灰度级如果区分成 2 组,数据如下:


  1. 0~3:4 个像素点,统计的像素个数是 19;

  2. 4~7:4 个像素点,统计的像素个数是 6。均衡化之后,灰度级如果区分成 2 组,数据如下:

  3. 0~3:4 个像素点,统计的像素个数是 11;

  4. 4~7:4 个像素点,统计的像素个数是 14。


如果希望得到更大范围的直方图,只需要将累计概率乘以更大的灰度级即可,具体可以自行尝试。

橡皮擦的小节

希望今天的 1 个小时你有所收获,我们下篇博客见~


网虫。博主 ID:梦想橡皮擦,希望大家<font color="red">点赞</font>、<font color="red">评论</font>、<font color="red">收藏</font>。

发布于: 2021 年 07 月 30 日阅读数: 4
用户头像

爬虫 100 例作者,蓝桥签约作者,博客专家 2021.02.06 加入

6 年产品经理+教学经验,3 年互联网项目管理经验; 互联网资深爱好者; 沉迷各种技术无法自拔,导致年龄被困在 25 岁; CSDN 爬虫 100 例作者。 个人公众号“梦想橡皮擦”。

评论

发布
暂无评论
Python OpenCV 图像处理之直方图相关知识细节,学点细的