那么直方图是什么?您可以将直方图视为图形或绘图,从而可以总体了解图像的强度分布。它是在X轴上具有像素值(不总是从0到255的范围),在Y轴上具有图像中相应像素数的图。
这只是理解图像的另一种方式。通过查看图像的直方图,您可以直观地了解该图像的对比度,亮度,强度分布等。当今几乎所有图像处理工具都提供直方图功能。以下是剑桥彩色网站的图片,我建议您访问该网站以获取更多详细信息。
您可以看到图像及其直方图。(请记住,此直方图是针对灰度图像而非彩色图像绘制的)。直方图的左侧区域显示图像中较暗像素的数量,而右侧区域则显示明亮像素的数量。从直方图中,您可以看到暗区域多于亮区域,而中间调的数量(中间值的像素值,例如127附近)则非常少。
寻找直方图现在我们有了一个关于直方图的想法,我们可以研究如何找到它。OpenCV和Numpy都为此内置了功能。在使用这些功能之前,我们需要了解一些与直方图有关的术语。
BINS:上面的直方图显示每个像素值的像素数,即从0到255。即,您需要256个值来显示上面的直方图。但是考虑一下,如果您不需要分别找到所有像素值的像素数,而是找到像素值间隔中的像素数怎么办? 例如,您需要找到介于0到15之间的像素数,然后找到16到31之间,...,240到255之间的像素数。只需要16个值即可表示直方图。这就是在OpenCV教程中有关直方图的示例中显示的内容。
因此,您要做的就是将整个直方图分成16个子部分,每个子部分的值就是其中所有像素数的总和。 每个子部分都称为“ BIN”。在第一种情况下,bin的数量为256个(每个像素一个),而在第二种情况下,bin的数量仅为16个。BINS由OpenCV文档中的histSize术语表示。
Dims:这是我们为其收集数据的参数的数量。在这种情况下,我们仅收集关于强度值的一件事的数据。所以这里是1。
RANGE:这是您要测量的强度值的范围。通常,它是[0,256]
,即所有强度值。
因此,现在我们使用cv.calcHist()函数查找直方图。让我们熟悉一下该函数及其参数:
因此,让我们从示例图像开始。只需以灰度模式加载图像并找到其完整直方图即可。
img = cv.imread('home.jpg',0)hist = cv.calcHist([img],[0],None,[256],[0,256])
hist是256x1的数组,每个值对应于该图像中具有相应像素值的像素数。
2. numpy的直方图计算Numpy还为您提供了一个函数np.histogram()。因此,除了calcHist()函数外,您可以尝试下面的代码:
hist,bins = np.histogram(img.ravel(),256,256])
hist与我们之前计算的相同。但是bin将具有257个元素,因为Numpy计算出bin的范围为0-0.99
、1-1.99
、2-2.99
等。因此最终范围为255-255.99
。为了表示这一点,他们还在最后添加了256。但是我们不需要256。最多255就足够了。
hist = np.bincount(img.ravel(),minlength = 256)
注意 OpenCV函数比np.histogram()快大约40倍。因此,尽可能使用OpenCV函数。
现在我们应该绘制直方图,但是怎么绘制?
绘制直方图有两种方法, 1. 简短的方法:使用Matplotlib绘图功能 2. 稍长的方法:使用OpenCV绘图功能
1. 使用MatplotlibMatplotlib带有直方图绘图功能:matplotlib.pyplot.hist()
它直接找到直方图并将其绘制。您无需使用calcHist()或np.histogram()函数来查找直方图。请参见下面的代码:
import numpy as np cv2 as cvfrom matplotlib pyplot as pltimg = cv.imread(或者,您可以使用matplotlib的法线图,这对于BGR图是很好的。为此,您需要首先找到直方图数据。试试下面的代码:
)color = (b',gr)for i,col in enumerate(color): histr = cv.calcHist([img],[i],256]) plt.plot(histr,color = col) plt.xlim([0,256])plt.show()结果:
您可以从上图中得出,蓝色在图像中具有一些高值域(显然这应该是由于天空)
2. 使用 OpenCV好吧,在这里您可以调整直方图的值及其bin值,使其看起来像x,y坐标,以便您可以使用cv.line()或cv.polyline()函数绘制它以生成与上述相同的图像。OpenCV-Python2官方示例已经提供了此功能。检查示例/python/hist.py中的代码。
掩码的应用我们使用了
cv.calcHist()
来查找整个图像的直方图。如果你想找到图像某些区域的直方图呢?只需创建一个掩码图像,在你要找到直方图为白色,否则黑色。然后把这个作为掩码传递。img = cv.imread(# create a maskmask = np.zeros(img.shape[:2],np.uint8)mask[100:300,100:400] = 255masked_img = cv.bitwise_and(img,img,mask = mask) 计算掩码区域和非掩码区域的直方图# 检查作为掩码的第三个参数hist_full = cv.calcHist([img],1)">])hist_mask= cv.calcHist([img],mask,1)">])plt.subplot(221),plt.imshow(img,1)">gray)plt.subplot(222),plt.imshow(mask,1)">)plt.subplot(223),plt.imshow(masked_img,1)">)plt.subplot(224),plt.plot(hist_full),plt.plot(hist_mask)plt.xlim([0,1)">])plt.show()
查看结果。在直方图中,蓝线表示完整图像的直方图,绿线表示掩码区域的直方图。
摘自:http://woshicver.com/FifthSection/4_10_1_%E7%9B%B4%E6%96%B9%E5%9B%BE-1%EF%BC%9A%E6%9F%A5%E6%89%BE%EF%BC%8C%E7%BB%98%E5%88%B6%EF%BC%8C%E5%88%86%E6%9E%90/
总结以上是内存溢出为你收集整理的【python opencv】直方图查找、绘制和分析全部内容,希望文章能够帮你解决【python opencv】直方图查找、绘制和分析所遇到的程序开发问题。
如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)