OpenCV
开源计算机视觉库
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Modules Pages
直方图均衡化

上一篇教程: 仿射变换
下一篇教程: 直方图计算

原作者Ana Huamán
兼容性OpenCV >= 3.0

目标

在本教程中,您将学习

  • 什么是图像直方图以及为什么它有用
  • 使用 OpenCV 函数 cv::equalizeHist 来均衡图像的直方图

理论

什么是图像直方图?

  • 它是图像强度分布的图形表示。
  • 它量化了每个强度值的像素数量。

什么是直方图均衡化?

  • 这是一种提高图像对比度的方法,目的是扩展强度范围(另请参见相应的 维基百科词条)。
  • 为了更清楚地说明,从上图可以看出,像素似乎聚集在可用强度范围的中间。直方图均衡化所做的是*扩展*此范围。请看下图:绿色圆圈表示*像素不足*的强度值。应用均衡化后,我们将得到中间图所示的直方图。右图显示了结果图像。

它是如何工作的?

  • 均衡化意味着将一种分布(给定的直方图)映射到另一种分布(强度值的更宽、更均匀的分布),以便强度值分布在整个范围内。
  • 为了实现均衡化效果,重新映射应该使用*累积分布函数 (CDF)*(更多细节,请参考*学习 OpenCV*)。对于直方图H(i),其*累积分布* H(i)

    H(i)=0j<iH(j)

    为了将其用作重新映射函数,我们必须对H(i)进行归一化,使其最大值为 255(或图像强度的最大值)。从上面的例子中,累积函数是

  • 最后,我们使用简单的重新映射过程来获得均衡化图像的强度值

    equalized(x,y)=H(src(x,y))

代码

  • 这段程序的功能是什么?
    • 加载图像
    • 将原始图像转换为灰度图像
    • 使用 OpenCV 函数 cv::equalizeHist 均衡直方图
    • 在一个窗口中显示源图像和均衡化后的图像。
  • 可下载代码: 点击 这里
  • 代码概览
    #include <iostream>
    using namespace cv;
    using namespace std;
    int main( int argc, char** argv )
    {
    CommandLineParser parser( argc, argv, "{@input | lena.jpg | input image}" );
    Mat src = imread( samples::findFile( parser.get<String>( "@input" ) ), IMREAD_COLOR );
    if( src.empty() )
    {
    cout << "无法打开或找到图像!\n" << endl;
    cout << "用法:" << argv[0] << " <输入图像>" << endl;
    return -1;
    }
    cvtColor( src, src, COLOR_BGR2GRAY );
    Mat dst;
    equalizeHist( src, dst );
    imshow( "源图像", src );
    imshow( "均衡化后的图像", dst );
    waitKey();
    return 0;
    }
    用于命令行解析。
    定义 utility.hpp:890
    n 维密集数组类
    定义 mat.hpp:829
    bool empty() const
    如果数组没有元素,则返回 true。
    std::string String
    定义 cvstd.hpp:151
    int main(int argc, char *argv[])
    定义 highgui_qt.cpp:3
    定义 core.hpp:107
    STL 命名空间。

解释

  • 加载源图像

    CommandLineParser parser( argc, argv, "{@input | lena.jpg | input image}" );
    Mat src = imread( samples::findFile( parser.get<String>( "@input" ) ), IMREAD_COLOR );
    if( src.empty() )
    {
    cout << "无法打开或找到图像!\n" << endl;
    cout << "用法:" << argv[0] << " <输入图像>" << endl;
    return -1;
    }
  • 将其转换为灰度图像

    cvtColor( src, src, COLOR_BGR2GRAY );
  • 使用函数cv::equalizeHist 应用直方图均衡化

    Mat dst;
    equalizeHist( src, dst );

    很容易看出,唯一的参数是原始图像和输出(均衡化)图像。

  • 显示两张图像(原始图像和均衡化图像)

    imshow( "源图像", src );
    imshow( "均衡化图像", dst );
  • 等待用户退出程序

结果

  1. 为了更好地欣赏均衡化的结果,让我们引入一张对比度不高的图像,例如

顺便说一句,它的直方图是这样的

注意像素集中在直方图的中心附近。

  1. 使用我们的程序应用均衡化后,我们得到以下结果

这张图像的对比度肯定更高。像这样查看它的新直方图

注意像素数量如何在强度范围内更均匀地分布。

注意
您想知道我们是如何绘制上面显示的直方图的?查看以下教程!