OpenCV-Python 直方图-1:查找、绘制和分析 | 二十六

摘要:
目标是学习使用OpenCV和Numpy函数来查找直方图。使用OpenCV和Matplotlib函数绘制直方图。您将看到以下函数:cv。calcHist()、np-Histogram()等。OpenCV和Numpy具有用于此目的的内置函数。这是OpenCV教程中直方图示例中显示的内容。BINS由OpenCV文档中的术语histSize表示。因此,尽可能使用OpenCV函数。官方的OpenCV-Python2示例已经提供了这个函数。

目标

学会

  • 使用OpenCV和Numpy函数查找直方图
  • 使用OpenCV和Matplotlib函数绘制直方图
  • 你将看到以下函数:cv.calcHist(),np.histogram()等。

理论

那么直方图是什么?您可以将直方图视为图形或绘图,从而可以总体了解图像的强度分布。它是在X轴上具有像素值(不总是从0到255的范围),在Y轴上具有图像中相应像素数的图。

这只是理解图像的另一种方式。通过查看图像的直方图,您可以直观地了解该图像的对比度,亮度,强度分布等。当今几乎所有图像处理工具都提供直方图功能。以下是剑桥彩色网站的图片,我建议您访问该网站以获取更多详细信息。

OpenCV-Python 直方图-1:查找、绘制和分析 | 二十六第1张

您可以看到图像及其直方图。(请记住,此直方图是针对灰度图像而非彩色图像绘制的)。直方图的左侧区域显示图像中较暗像素的数量,而右侧区域则显示明亮像素的数量。从直方图中,您可以看到暗区域多于亮区域,而中间调的数量(中间值的像素值,例如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],即所有强度值。

1. OpenCV中的直方图计算

因此,现在我们使用cv.calcHist()函数查找直方图。让我们熟悉一下该函数及其参数:

cv.calcHist(images,channels,mask,histSize,ranges [,hist [,accumulate]])
  1. images:它是uint8或float32类型的源图像。它应该放在方括号中,即“ [img]”。
  2. channels:也以方括号给出。它是我们计算直方图的通道的索引。例如,如果输入为灰度图像,则其值为[0]。对于彩色图像,您可以传递[0],[1]或[2]分别计算蓝色,绿色或红色通道的直方图。
  3. mask:图像掩码。为了找到完整图像的直方图,将其指定为“无”。但是,如果要查找图像特定区域的直方图,则必须为此创建一个掩码图像并将其作为掩码。(我将在后面显示一个示例。)
  4. histSize:这表示我们的BIN计数。需要放在方括号中。对于全尺寸,我们通过[256]。
  5. ranges:这是我们的RANGE。通常为[0,256]。

因此,让我们从示例图像开始。只需以灰度模式加载图像并找到其完整直方图即可。

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,[0,256])

hist与我们之前计算的相同。但是bin将具有257个元素,因为Numpy计算出bin的范围为0-0.991-1.992-2.99等。因此最终范围为255-255.99。为了表示这一点,他们还在最后添加了256。但是我们不需要256。最多255就足够了。

  • 另外
    Numpy还有另一个函数np.bincount(),它比np.histogram()快10倍左右。因此,对于一维直方图,您可以更好地尝试一下。不要忘记在np.bincount中设置minlength = 256。例如,hist = np.bincount(img.ravel(),minlength = 256)

注意
OpenCV函数比np.histogram()快大约40倍。因此,尽可能使用OpenCV函数。

现在我们应该绘制直方图,但是怎么绘制?

绘制直方图

有两种方法,

  1. 简短的方法:使用Matplotlib绘图功能
  2. 稍长的方法:使用OpenCV绘图功能

1. 使用Matplotlib

Matplotlib带有直方图绘图功能:matplotlib.pyplot.hist()
它直接找到直方图并将其绘制。您无需使用calcHist()或np.histogram()函数来查找直方图。请参见下面的代码:

import numpy as np
import cv2 as cv
from matplotlib import pyplot as plt
img = cv.imread('home.jpg',0)
plt.hist(img.ravel(),256,[0,256]); plt.show()

你将得到如下的结果:
OpenCV-Python 直方图-1:查找、绘制和分析 | 二十六第2张

或者,您可以使用matplotlib的法线图,这对于BGR图是很好的。为此,您需要首先找到直方图数据。试试下面的代码:

import numpy as np
import cv2 as cv
from matplotlib import pyplot as plt
img = cv.imread('home.jpg')
color = ('b','g','r')
for i,col in enumerate(color):
    histr = cv.calcHist([img],[i],None,[256],[0,256])
    plt.plot(histr,color = col)
    plt.xlim([0,256])
plt.show()

结果:
OpenCV-Python 直方图-1:查找、绘制和分析 | 二十六第3张

您可以从上图中得出,蓝色在图像中具有一些高值域(显然这应该是由于天空)

2. 使用 OpenCV

好吧,在这里您可以调整直方图的值及其bin值,使其看起来像x,y坐标,以便您可以使用cv.line()或cv.polyline()函数绘制它以生成与上述相同的图像。OpenCV-Python2官方示例已经提供了此功能。检查示例/python/hist.py中的代码。

掩码的应用

我们使用了cv.calcHist()来查找整个图像的直方图。如果你想找到图像某些区域的直方图呢?只需创建一个掩码图像,在你要找到直方图为白色,否则黑色。然后把这个作为掩码传递。

img = cv.imread('home.jpg',0)
# create a mask
mask = np.zeros(img.shape[:2], np.uint8)
mask[100:300, 100:400] = 255
masked_img = cv.bitwise_and(img,img,mask = mask)
# 计算掩码区域和非掩码区域的直方图
# 检查作为掩码的第三个参数
hist_full = cv.calcHist([img],[0],None,[256],[0,256])
hist_mask = cv.calcHist([img],[0],mask,[256],[0,256])
plt.subplot(221), plt.imshow(img, 'gray')
plt.subplot(222), plt.imshow(mask,'gray')
plt.subplot(223), plt.imshow(masked_img, 'gray')
plt.subplot(224), plt.plot(hist_full), plt.plot(hist_mask)
plt.xlim([0,256])
plt.show()

查看结果。在直方图中,蓝线表示完整图像的直方图,绿线表示掩码区域的直方图。

OpenCV-Python 直方图-1:查找、绘制和分析 | 二十六第4张

附加资源

  1. Cambridge in Color website:http://www.cambridgeincolour.com/tutorials/histograms1.htm

欢迎关注磐创博客资源汇总站:
http://docs.panchuang.net/

欢迎关注PyTorch官方中文教程站:
http://pytorch.panchuang.net/

OpenCV中文官方文档:
http://woshicver.com/

免责声明:文章转载自《OpenCV-Python 直方图-1:查找、绘制和分析 | 二十六》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇Oracle配置远程连接,添加新的IP地址,必须保存原来的127.0.0.1,否则都不能连接RAID详解[RAID0/RAID1/RAID5]下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

OpenCV+python轮廓

OpenCV中的轮廓 1.1什么是轮廓轮廓可以简单认为成连续的点(连着边界)连在一起的曲线,具有相同的颜色或者灰度。轮廓在形状分析和物体的检测和识别中很有用。 为了准确,要使用二值化图像。需要进行阀值化处理或者Canny边界检测。 查找轮廓的函数会修改原始图像。如果之后想继续使用原始图像,应该将原始图像储存到其他变量中。 在OpenCV中,查找轮廓...

opencv获取当前帧数据问题

在用opencv打开摄像头,获取当前帧数据,处理图像人脸识别、操作硬件等动作之后再获取当前帧数据。出现如下问题: 摄像头的每帧数据会全部存入缓冲区,需要一帧接一帧的把缓冲区的图片依次进行处理,但处理速度小于存进缓冲区的速度,那么缓冲区的图片就越积累越多,会导致处理内容都不是实时的、导致延时现象很明显。 解决办法:增加一个线程,循环获取摄像头数据,及时清空缓...

java 实现基于opencv全景图合成

因项目需要,自己做了demo,从中学习很多,所以分享出来,希望有这方面需求的少走一些弯路,opencv怎么安装网上教程多多,这里不加详细说明,我安装的opencv-3.3.0  如上图所示,找到相应的jar包,这里讲一下如何这个jar如何导入Maven仓库 mvn install:install-file -Dfile=D:opencv-3.0.0ope...

pkg-config 用法

在进行使用fuse 2.9写程序的时候,遇到了pkg-config 命令和 .pc 文件。本篇博客就具体说明一下pkg-config 命令是什么? 我们首先看一下: gcc -Wall hello.c `pkg-config fuse3 --cflags --libs` -o hello [root@docker pkgconfig]# cat fuse....

OpenCV实现人脸识别

// 总结:OpenCV实现人脸识别// 2016.2.12 by Huangtao//  主要有以下步骤:1、人脸检测2、人脸预处理3、从收集的人脸训练机器学习算法4、人脸识别5、收尾工作 =================================================人脸检测算法:===========================...

系统综合实践 第5次实践作业

目录 (一)Python容器文件配置 (二)程序的部署运行 (1)helloworld (2)日历输出 (3)数据库操作 (4)opencv (三)实验感想 作业地址 (一)Python容器文件配置 目录结构 Dockerfile FROM python MAINTAINER yezi WORKDIR /app #添加依赖文件 C...