Opencv-Python学习笔记(十一):图像直方图

本篇博客记录学习OpenCV-Python图像直方图的相关知识。

  • 使用OpenCV和Numpy函数查找直方图,使用Matplotlib函数绘制直方图
  • 您要学习的函数有:cv2.calcHist()np.histogram()等。

原理

什么是直方图呢?通过直方图我们可以对整幅图像的灰度分布有一个整体的了解。直方图的 x 轴是灰度值( 0 到 255)y 轴是图片中具有同一个灰度值的点的数目
直方图其实就是对图像的另一种解释。以下图为例,通过直方图我们可以对图像的对比度,亮度,灰度分布等有一个直观的认识。几乎所有的图像处理软件都提供了直方图分析功能。

统计、绘制直方图

函数 cv2.calcHist ()可以帮助我们统计一幅图像的直方图。
cv2.calcHist(images; channels; mask; histSize; ranges[; hist[; accumulate]])
1. images: 原图像(图像格式为 uint8 或 float32)。当传入函数时应该用中括号 [] 括起来,例如: [img]。
2. channels: 同样需要用中括号括起来,它会告诉函数我们要统计那幅图像的直方图。如果输入图像是灰度图,它的值就是 [0];如果是彩色图像的话,传入的参数可以是 [0], [1], [2] 它们分别对应着通道 B, G, R。
3. mask: 掩模图像。要统计整幅图像的直方图就把它设为 None。但是如果你想统计图像某一部分的直方图的话,你就需要制作一个掩模图像,并使用它。(后边有例子)
4. histSize:BIN 的数目。也应该用中括号括起来,例如: [256]。
5. ranges: 像素值范围,通常为 [0,256]。

有两种方法来绘制直方图:
1. Short Way(简单方法):使用 Matplotlib 中的绘图函数。
2. Long Way(复杂方法):使用 OpenCV 绘图函数。

使用 Matplotlib Matplotlib 中有直方图绘制函数: matplotlib.pyplot.hist()它可以直接统计并绘制直方图。

接下来我们以下图为例进行操作。

                                                               

 

# -*- coding: utf-8 -*-
# @Time    : 2019/11/10 19:33
# @Author  : MMagicLoren
# @Email   : 993983320@qq.com
# @File    : 直方图.py
# @Software: PyCharm
import cv2 as cv
import matplotlib.pyplot as plt
import numpy as np


def plot_demo(image):
    plt.hist(image.ravel(), 256, [0, 256])  # image.ravel()是将多维数组降为一维数组,256为bins数量,[0, 256]为范围
    plt.show()


def image_hist(image):
    color = ('blue', 'green', 'red')
    for i, color in enumerate(color):
        # 计算出直方图,calcHist(images, channels, mask, histSize(有多少个bin), ranges[, hist[, accumulate]]) -> hist
        # hist 是一个 256x1 的数组,每一个值代表了与该灰度值对应的像素点数目。
        hist = cv.calcHist(image, [i], None, [256], [0, 256])
        print(hist.shape)
        plt.plot(hist, color=color)
        plt.xlim([0, 256])
    plt.show()


if __name__ == '__main__':
    src = cv.imread("F:/Pycharm/opencv_exercises-master/images/Mimi256.jpg")  # 读入图片放进src中
    cv.namedWindow("input image")  # 创建窗口
    cv.imshow("input image", src)  # 将src图片放入该创建的窗口中
    plot_demo(src)
    image_hist(src)
    cv.waitKey(0)  # 等有键输入或者1000ms后自动将窗口消除,0表示只用键输入结束窗口
    cv.destroyAllWindows()  # 关闭所有窗口

运行结果图:

使用掩模

要统计图像某个局部区域的直方图只需要构建一副掩模图像。将要统计的部分设置成白色,其余部分为黑色,就构成了一副掩模图像。然后把这个掩模图像传给函数就可以了。
 

import cv2 as cv
import matplotlib.pyplot as plt
import numpy as np


def mask_process(image):
    image = cv.cvtColor(image, cv.COLOR_BGR2RGB)
    mask = np.zeros(image.shape[:2], np.uint8)
    mask[0:150, 60:200] = 255
    # cv.imshow("mask", mask)
    masked_img = cv.bitwise_and(image, image, mask=mask)
    # cv.imshow("masked_image", masked_img)
    hist_full = cv.calcHist([image], [0], None, [256], [0, 256])
    hist_mask = cv.calcHist([image], [0], mask, [256], [0, 256])
    plt.subplot(221), plt.imshow(image, '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()


if __name__ == '__main__':
    src = cv.imread("F:/Pycharm/opencv_exercises-master/images/Mimi256.jpg")  # 读入图片放进src中
    cv.namedWindow("input image")  # 创建窗口
    cv.imshow("input image", src)  # 将src图片放入该创建的窗口中
    mask_process(src)
    cv.waitKey(0)  # 等有键输入或者1000ms后自动将窗口消除,0表示只用键输入结束窗口
    cv.destroyAllWindows()  # 关闭所有窗口

效果图:

直方图均衡化

原理

想象一下如果一副图像中的大多是像素点的像素值都集中在一个像素值范围之内会怎样呢?例如,如果一幅图片整体很亮,那所有的像素值应该都会很高。但是一副高质量的图像的像素值分布应该很广泛。所以应该把它的直方图做一个横向拉伸(如下图),这就是直方图均衡化要做的事情。通常情况下这种操作会改善图像的对比度。


                                                              

 

# -*- coding: utf-8 -*-
# @Time    : 2019/11/10 19:33
# @Author  : MMagicLoren
# @Email   : 993983320@qq.com
# @File    : 直方图.py
# @Software: PyCharm
import cv2 as cv
import matplotlib.pyplot as plt
import numpy as np


def equalHist_demo(image):
    gray = cv.cvtColor(image, cv.COLOR_BGR2GRAY)
    # 全局直方图均衡化,用于增强图像对比度,即黑的更黑,白的更白
    dst = cv.equalizeHist(gray)
    cv.imshow("equalHist_demo", dst)
  

if __name__ == '__main__':
    src = cv.imread("F:/Pycharm/opencv_exercises-master/images/rice.png")  # 读入图片放进src中
    cv.namedWindow("input image")  # 创建窗口
    cv.imshow("input image", src)  # 将src图片放入该创建的窗口中
    equalHist_demo(src)
    cv.waitKey(0)  # 等有键输入或者1000ms后自动将窗口消除,0表示只用键输入结束窗口
    cv.destroyAllWindows()  # 关闭所有窗口

我们可以看出对比度明显增强。

但是有时候我们这样整体处理一幅图像会丢失一些信息,这个时候我们就需要自适应的直方图均衡化。这种情况下,整幅图像会被分成很多小块,这些小块被称为“tiles”(在 OpenCV 中 tiles 的大小默认是 8x8),然后再对每一个小块分别进行直方图均衡化(跟前面类似)。所以在每一个的区域中,直方图会集中在某一个小的区域中(除非有噪声干扰)。如果有噪声的话,噪声会被放大。为了避免这种情况的出现要使用对比度限制。对于每个小块来说,如果直方图中的 bin 超过对比度的上限的话,就把其中的像素点均匀分散到其他 bins 中,然后在进行直方图均衡化。最后,为了去除每一个小块之间“人造的”(由于算法造成)边界,再使用双线性差值,对小块进行缝合。

# -*- coding: utf-8 -*-
# @Time    : 2019/11/10 19:33
# @Author  : MMagicLoren
# @Email   : 993983320@qq.com
# @File    : 直方图.py
# @Software: PyCharm
import cv2 as cv
import matplotlib.pyplot as plt
import numpy as np


def equalHist_demo(image):
    gray = cv.cvtColor(image, cv.COLOR_BGR2GRAY)
    # 局部直方图均衡化
    clahe = cv.createCLAHE(clipLimit=2.0, tileGridSize=(8, 8))
    clahe_dst = clahe.apply(gray)
    cv.imshow("clahe", clahe_dst)
  

if __name__ == '__main__':
    src = cv.imread("F:/Pycharm/opencv_exercises-master/images/rice.png")  # 读入图片放进src中
    cv.namedWindow("input image")  # 创建窗口
    cv.imshow("input image", src)  # 将src图片放入该创建的窗口中
    equalHist_demo(src)
    cv.waitKey(0)  # 等有键输入或者1000ms后自动将窗口消除,0表示只用键输入结束窗口
    cv.destroyAllWindows()  # 关闭所有窗口

结果如图所示:

直方图比较

建议去看看贾志刚老师的课程。我也不是很懂,emmmm  直接上代码看一下效果吧。

# -*- coding: utf-8 -*-
# @Time    : 2019/11/10 19:33
# @Author  : MMagicLoren
# @Email   : 993983320@qq.com
# @File    : 直方图.py
# @Software: PyCharm
import cv2 as cv
import matplotlib.pyplot as plt
import numpy as np


# 创建直方图
def create_rgb_demo(image):
    h, w, c = image.shape
    rgbHist = np.zeros([16*16*16, 1], np.float32)
    bsize = 256 / 16
    for row in range(h):
        for col in range(w):
            b = image[row, col, 0]
            g = image[row, col, 1]
            r = image[row, col, 2]
            index = np.int(b/bsize)*16*16 + np.int(g/bsize)*16 + np.int(r/bsize)
            rgbHist[np.int(index), 0] = rgbHist[np.int(index), 0] + 1
    return rgbHist


# 利用直方图比较相似性,用巴氏和相关性比较好
def hist_compare(image_1, image_2):
    hist1 = create_rgb_demo(image1)
    hist2 = create_rgb_demo(image2)
    match1 = cv.compareHist(hist1, hist2, method=cv.HISTCMP_BHATTACHARYYA)
    match2 = cv.compareHist(hist1, hist2, method=cv.HISTCMP_CORREL)
    match3 = cv.compareHist(hist1, hist2, method=cv.HISTCMP_CHISQR)
    print("巴式距离:%s, 相关性:%s, 卡方:%s" % (match1, match2, match3))


if __name__ == '__main__':
    image1 = cv.imread("F:/Pycharm/opencv_exercises-master/images/rice.png")
    image2 = cv.imread("F:/Pycharm/opencv_exercises-master/images/noise_rice.png")
    cv.imshow("image1", image1)
    cv.imshow("image2", image2)
    hist_compare(image1, image2)
    cv.waitKey(0)  # 等有键输入或者1000ms后自动将窗口消除,0表示只用键输入结束窗口
    cv.destroyAllWindows()  # 关闭所有窗口

通过数值,我们可以很容易的判断图像的相似度。

完整工程代码:

# -*- coding: utf-8 -*-
# @Time    : 2019/11/10 19:33
# @Author  : MMagicLoren
# @Email   : 993983320@qq.com
# @File    : 直方图.py
# @Software: PyCharm
import cv2 as cv
import matplotlib.pyplot as plt
import numpy as np


def plot_demo(image):
    plt.hist(image.ravel(), 256, [0, 256])  # image.ravel()是将多维数组降为一维数组,256为bins数量,[0, 256]为范围
    plt.show()


def image_hist(image):
    color = ('blue', 'green', 'red')
    for i, color in enumerate(color):
        # 计算出直方图,calcHist(images, channels, mask, histSize(有多少个bin), ranges[, hist[, accumulate]]) -> hist
        # hist 是一个 256x1 的数组,每一个值代表了与该灰度值对应的像素点数目。
        hist = cv.calcHist(image, [i], None, [256], [0, 256])
        print(hist.shape)
        plt.plot(hist, color=color)
        plt.xlim([0, 256])
    plt.show()


def mask_process(image):
    image = cv.cvtColor(image, cv.COLOR_BGR2RGB)
    mask = np.zeros(image.shape[:2], np.uint8)
    mask[0:150, 60:200] = 255
    # cv.imshow("mask", mask)
    masked_img = cv.bitwise_and(image, image, mask=mask)
    # cv.imshow("masked_image", masked_img)
    hist_full = cv.calcHist([image], [0], None, [256], [0, 256])
    hist_mask = cv.calcHist([image], [0], mask, [256], [0, 256])
    plt.subplot(221), plt.imshow(image, '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()


def equalHist_demo(image):
    gray = cv.cvtColor(image, cv.COLOR_BGR2GRAY)
    # 全局直方图均衡化,用于增强图像对比度,即黑的更黑,白的更白
    # dst = cv.equalizeHist(gray)
    # cv.imshow("equalHist_demo", dst)
    # 局部直方图均衡化
    clahe = cv.createCLAHE(clipLimit=2.0, tileGridSize=(8, 8))
    clahe_dst = clahe.apply(gray)
    cv.imshow("clahe", clahe_dst)


# 创建直方图
def create_rgb_demo(image):
    h, w, c = image.shape
    rgbHist = np.zeros([16*16*16, 1], np.float32)
    bsize = 256 / 16
    for row in range(h):
        for col in range(w):
            b = image[row, col, 0]
            g = image[row, col, 1]
            r = image[row, col, 2]
            index = np.int(b/bsize)*16*16 + np.int(g/bsize)*16 + np.int(r/bsize)
            rgbHist[np.int(index), 0] = rgbHist[np.int(index), 0] + 1
    return rgbHist


# 利用直方图比较相似性,用巴氏和相关性比较好
def hist_compare(image_1, image_2):
    hist1 = create_rgb_demo(image1)
    hist2 = create_rgb_demo(image2)
    match1 = cv.compareHist(hist1, hist2, method=cv.HISTCMP_BHATTACHARYYA)
    match2 = cv.compareHist(hist1, hist2, method=cv.HISTCMP_CORREL)
    match3 = cv.compareHist(hist1, hist2, method=cv.HISTCMP_CHISQR)
    print("巴式距离:%s, 相关性:%s, 卡方:%s" % (match1, match2, match3))


if __name__ == '__main__':
    # src = cv.imread("F:/Pycharm/opencv_exercises-master/images/rice.png")  # 读入图片放进src中
    # cv.namedWindow("input image")  # 创建窗口
    # cv.imshow("input image", src)  # 将src图片放入该创建的窗口中
    # plot_demo(src)
    # image_hist(src)
    # mask_process(src)
    # equalHist_demo(src)
    image1 = cv.imread("F:/Pycharm/opencv_exercises-master/images/Crystal256.jpg")
    image2 = cv.imread("F:/Pycharm/opencv_exercises-master/images/Mimi256.jpg")

    # create_rgb_demo(image1)
    cv.imshow("image1", image1)
    cv.imshow("image2", image2)
    hist_compare(image1, image2)

    cv.waitKey(0)  # 等有键输入或者1000ms后自动将窗口消除,0表示只用键输入结束窗口
    cv.destroyAllWindows()  # 关闭所有窗口

 

  • 7
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
第一课:绘制直方图 import cv2 as cv import numpy as np from matplotlib import pyplot as plt def plot_demo(image): plt.hist(image.ravel(),256,[0,256]) plt.show() def image_hist(image): color = ('blue','green','red') for i,color in enumerate(color): #计算直方图 hist = cv.calcHist(image,[i],None,[256],[0,256]) #画出直方图 plt.plot(hist,color=color) plt.xlim(0,256) plt.show() src = cv.imread("E:/opencv/picture/Greatwall.jpg") cv.imshow("initial_window",src) plot_demo(src) image_hist(src) cv.waitKey(0) cv.destroyAllWindows() 效果图: 7-1源图像 7-2 plot_demo 7-3 image_hist 分析: 1. 什么叫直方图? 答:简单来说就是图像中每个像素值的个数统计。比如说一副灰度图中像素值为0的有多少个,像素值为1的有多少个….直方图是一种分析图片的手段。 在计算直方图之前,有几个术语需要先了解一下: dims:要计算的通道数,对于灰度图dims = 1,对于rgb图像dims= 3 range:要计算的像素值范围,一般为【0,256】(不包括256) bins:子区间数目,如果我们统计0~255每个像素值,bins =256;如果划分区间,比如【0,15】,【16,31】,….【240,255】这样的16个区间,bins = 16 2. 需要安装matplotlib pip install matplotlib 3. plot_demo(matplotlib自带的计算并绘制直方图) def plot_demo(image): plt.hist(image.ravel(),256,[0,256]) plt.show() 1) image.ravel()函数的功能是将多维数组降为一维数组 2)matplotlib.pyplot.hist函数主要是计算直方图 hist函数原型:hist(x, bins=None, range=None,….) x参数表示是一个数组或一个序列,是指定每个bin(箱子)分布的数据 bins参数表示指定bin(箱子)的个数,也就是总共有几条条状图 range参数表示箱子的下限和上限。即横坐标显示的范围,范围之外的将被舍弃。 range这个参数一般都是[0,256] 3)plt.show()显示直方图 4. image_hist def image_hist(image): #注意这个是tuple()哦,且里面的是字符‘’而不是字符串 color = ('blue','green','red') for i,color in enumerate(color): hist =cv.calcHist(image,[i],None,[256],[0,256]) plt.plot(hist,color=color) plt.xlim(0,256) plt.show() 1) enumerate() 函数用于将一个可遍历的数据对象(如列表、元组或字符串)组合为一个索引序列,同时列出数据下标和数据,一般用在 for 循环当中。 语法: enumerate(sequence, [start=0]) 参数: • sequence -- 一个序列、迭代器或其他支持迭代对象。 • start -- 下标起始位置。 返回值 返回 enumerate(枚举) 对象。 举个例子: 对于普通的for循环: >>>i = 0 >>> seq = ['one', 'two', 'three'] >>> for element in seq: print i, seq[i] i +=1 0 one 1 two 2 three 对于for循环使用enumerate: >>>seq = ['one', 'two', 'three'] >>> for i, element in enumerate(seq): print i, element 0 one 1 two 2 three 2)cv2.calcHist的原型为:calcHist(images, channels, mask, histSize, ranges[, hist[, accumulate]]) -> hist images参数表示输入图像,传入时应该用中括号[ ]括起来 channels参数表示传入图像的通道,如果是灰度图像,那就不用说了,只有一个通道,值为0,如果是彩色图像(有3个通道),那么值为0,1,2,中选择一个,对应着BGR各个通道。这个值也得用[ ]传入。 mask参数表示掩膜图像。如果统计整幅图,那么为None。主要是如果要统计部分图的直方图,就得构造相应的掩膜来计算。 histSize参数表示灰度级的个数,需要中括号,比如[256] ranges参数表示像素值的范围,通常[0,256]。此外,假如channels为[0,1],ranges为[0,256,0,180],则代表0通道范围是0-256,1通道范围0-180。 3)plt.plot与plt.xlim函数 plt.plot(hist,color=color) plt.xlim(0,256) plt.plot函数:绘制函数图像 参数一:列表或者数组 参数二:控制颜色参数 plt.xlim函数:设置坐标轴刻度的取值范围 第二课 直方图均衡化与比较 均衡化(全局与部分) opencv直方图均衡化都是基于灰度图像的 作用:直方图均衡化可以调整图像的对比度,使图像更加清晰,对比度增强。是图像增强的一个手段。 直方图均衡化:如果一副图像的像素占有很多的灰度级而且分布均匀,那么这样的图像往往有高对比度和多变的灰度色调。直方图均衡化就是一种能仅靠输入图像直方图信息自动达到这种效果的变换函数。它的基本思想是对图像中像素个数多的灰度级进行展宽,而对图像中像素个数少的灰度进行压缩,从而扩展像元取值的动态范围,提高了对比度和灰度色调的变化,使图像更加清晰。 源码: import cv2 as cv import numpy as np from matplotlib import pyplot as plt def hist_demo(image): hist = plt.hist(image.ravel(),256,[0,256]) plt.show() def Image_hist(image): color = ("b","g","r") for i,color in enumerate(color): #计算直方图 hist = cv.calcHist(image,[i],None,[256],[0,256]) plt.xlim(0,256) plt.plot(hist,color) plt.show() #opencv直方图均衡化都是基于灰度图像的 #直方图均衡化可以自动的调整图像的对比度,使图像更加清晰化,对比度增强,是图像增强的一个手段 def equalHist_demo(image): gray = cv.cvtColor(image,cv.COLOR_BGR2GRAY) cv.imshow("gray",gray) dst = cv.equalizeHist(gray) cv.imshow("dst_win1",dst) #可以明显的看出dst_win1的对比度要比gray的对比度要高好多。 def clahe_demo(image): gray = cv.cvtColor(image, cv.COLOR_BGR2GRAY) cv.imshow("gray", gray) clahe = cv.createCLAHE(clipLimit=5.0,tileGridSize=(8,8)) dst = clahe.apply(gray) cv.imshow("dst_win2", dst) #之前学习的调节对比度以及锐化图像的方法 def contrast_demo(image,c,gamm): #blank = np.zeros(image.shape,np.uint8) #dst = cv.addWeighted(image,c,blank,0,gamm) kernel = np.array([[0,-1,0],[-1,5,-1],[0,-1,0]]) dst = cv.filter2D(image,-1,kernel) cv.imshow("dst_win",dst) src = cv.imread("E:/opencv/picture/lena.jpg") cv.imshow("inital_win",src) #hist_demo(src) #Image_hist(src) #contrast_demo(src,2,2) equalHist_demo(src) cv.waitKey(0) cv.destroyAllWindows() 得出结果如下: 原图 灰度图 图像全均衡化 自适应直方图均衡化(均衡参数可调推荐) 分析: 1)#opencv直方图均衡化都是基于灰度图像的 #直方图均衡化可以自动的调整图像的对比度,使图像更加清晰化,对比度增强,是图像增强的一个手段 #全局直方图均衡化 def equalHist_demo(image): gray = cv.cvtColor(image,cv.COLOR_BGR2GRAY) cv.imshow("gray",gray) dst = cv.equalizeHist(gray) cv.imshow("dst_win1",dst) #可以明显的看出dst_win1的对比度要比gray的对比度要高好多。 api: cv.equalizeHist(src) 对输入的源灰度图像进行直方图均衡化处理,从而提高图像质量。如果源图像是彩色图像,得需先转化成灰度图像。 2) 自适应(局部)直方图均衡化(可调参数) .全局直方图均衡化可能得到是一种全局意义上的均衡化,但是有的时候这种操作并不是很好,会把某些不该调整的部分给调整了。Opencv中还有一种直方图均衡化,它是一种局部直方图均衡化,也就是是说把整个图像分成许多小块(比如按10*10作为一个小块),那么对每个小块进行均衡化。 注:全图的直方图均衡化会导致对比度过度增强,所以在一些情况下应使用局部直方图均衡化; #局部直方图均衡化 def clahe_demo(image): gray = cv.cvtColor(image, cv.COLOR_BGR2GRAY) cv.imshow("gray", gray) clahe = cv

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

MMagicLoren

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值