颜色直方图


首先,先介绍一些Hist的基本使用。

Ref:【OpenCV】数字图像灰度直方图

官方文档:https://docs.opencv.org/trunk/d8/dbc/tutorial_histogram_calculation.html

不错博文:利用OpenCV的calcHist绘制灰度直方图、H-S直方图、BGR直方图和自定义直方图的源码及说明

From: compare histograms of grayscale images in opencv

  1. #include <opencv2/opencv.hpp>
  2.  
  3. void show_histogram(std::string const& name, cv::Mat1b const& image)
  4. {
  5. // Set histogram bins count
  6. int bins = ;
  7. int histSize[] = {bins};
  8. // Set ranges for histogram bins
  9. float lranges[] = {, };
  10. const float* ranges[] = {lranges};
  11. // create matrix for histogram
  12. cv::Mat hist;
  13. int channels[] = {};
  14.  
  15. // create matrix for histogram visualization
  16. int const hist_height = ;
  17. cv::Mat3b hist_image = cv::Mat3b::zeros(hist_height, bins);
  18.  
  19. cv::calcHist(&image, , channels, cv::Mat(), hist, , histSize, ranges, true, false);
  20.  
  21. double max_val=;
  22. minMaxLoc(hist, , &max_val);
  23.  
  24. // visualize each bin
  25. for(int b = ; b < bins; b++) {
  26. float const binVal = hist.at<float>(b);
  27. int const height = cvRound(binVal*hist_height/max_val);
  28. cv::line
  29. ( hist_image
  30. , cv::Point(b, hist_height-height), cv::Point(b, hist_height)
  31. , cv::Scalar::all()
  32. );
  33. }
  34. cv::imshow(name, hist_image);
  35. }
  36.  
  37. int main (int argc, const char* argv[])
  38. {
  39. // here you can use cv::IMREAD_GRAYSCALE to load grayscale image, see image2
  40. cv::Mat3b const image1 = cv::imread("C:\\workspace\\horse.png", cv::IMREAD_COLOR);
  41. cv::Mat1b image1_gray;
  42. cv::cvtColor(image1, image1_gray, cv::COLOR_BGR2GRAY);
  43. cv::imshow("image1", image1_gray);
  44. show_histogram("image1 hist", image1_gray);
  45.  
  46. cv::Mat1b const image2 = cv::imread("C:\\workspace\\bunny.jpg", cv::IMREAD_GRAYSCALE);
  47. cv::imshow("image2", image2);
  48. show_histogram("image2 hist", image2);
  49.  
  50. cv::waitKey();
  51. return ;
  52. }

Histogram equalization


参考:笑得很甜

偏差、增益参数

线性混合算子

非线性算子

覆盖算子


直方图均衡化

直方图的观看规则就是“左黑右白”,左边代表暗部,右边代表亮部,而中间则代表中间调。 纵向上的高度代表像素密集程度,越高,代表的就是分布在这个亮度上的像素很多。

如何看懂照片的直方图?

对比度

对比度在直方图上的体现就是高光和阴影部分都有像素。
它可以很少,但是必须有,否则照片看起来就很灰了。正常照片变为低对比度后的直方图对比:
                     

  当然,也有看起来不错的低对比度图片,但它的直方图不会像上图那样极端,一般都是没有纯黑,但高光都比较足:
  

均衡化

(a) vs (e)  对比度有所加强。

However,可能放大暗区域的噪声。

  1. 它的基本思想是对图像中像素个数多的灰度级进行展宽,而对图像中像素个数少的灰度进行压缩,从而扩展像原取值的动态范围,提高了对比度和灰度色调的变化,使图像更加清晰。
  2.  
  3. http://blog.csdn.net/xiaowei_cqu/article/details/7606607
    从分布图上的理解就是希望原始图像中y轴的值在新的分布中尽可能的展开。
    变换过程是利用累积分布函数对原始分布进行映射,生成新的均匀拉伸的分布。
    因此,对应每个点的操作是寻找原始分布中y值在均匀分布中的位置,如下图是理想的单纯高斯分布映射的示意图:

  1. int main()
  2. {
  3. IplImage * image= cvLoadImage("baboon.jpg");
  4. //显示原图及直方图
  5. myShowHist("Source",image);
  6.  
  7. IplImage* eqlimage=cvCreateImage(cvGetSize(image),image->depth,);
  8. //分别均衡化每个信道
  9. IplImage* redImage=cvCreateImage(cvGetSize(image),image->depth,);
  10. IplImage* greenImage=cvCreateImage(cvGetSize(image),image->depth,);
  11. IplImage* blueImage=cvCreateImage(cvGetSize(image),image->depth,);
  12. cvSplit(image,blueImage,greenImage,redImage,NULL);
  13.  
  14. cvEqualizeHist(redImage,redImage);
  15. cvEqualizeHist(greenImage,greenImage);
  16. cvEqualizeHist(blueImage,blueImage);
  17. //均衡化后的图像
  18. cvMerge(blueImage,greenImage,redImage,NULL,eqlimage);
  19. myShowHist("Equalized",eqlimage);
  20. }

cvEqualizeHist

此函数只能处理单通道的灰色图像,对于彩色图像,我们可以把每个信道分别均衡化,再Merge为彩色图像。

算法演示:

自定义均衡化

(直方图匹配 or 直方图规定化)

  1. //将图像与特定函数分布histv[]匹配
  2. void myHistMatch(IplImage *img,double histv[])
  3. {
  4. int bins = ;
  5. int sizes[] = {bins};
  6. CvHistogram *hist = cvCreateHist(,sizes,CV_HIST_ARRAY);
  7. cvCalcHist(&img,hist);
  8. cvNormalizeHist(hist,);
  9. double val_1 = 0.0;
  10. double val_2 = 0.0;
  11. uchar T[] = {};
  12. double S[] = {};
  13. double G[] = {};
  14. for (int index = ; index<; ++index)
  15. {
  16. val_1 += cvQueryHistValue_1D(hist,index);
  17. val_2 += histv[index];
  18. G[index] = val_2;
  19. S[index] = val_1;
  20. }
  21.  
  22. double min_val = 0.0;
  23. int PG = ;
  24. for ( int i = ; i<; ++i)
  25. {
  26. min_val = 1.0;
  27. for(int j = ;j<; ++j)
  28. {
  29. if( (G[j] - S[i]) < min_val && (G[j] - S[i]) >= )
  30. {
  31. min_val = (G[j] - S[i]);
  32. PG = j;
  33. }
  34.  
  35. }
  36. T[i] = (uchar)PG;
  37. }
  38.  
  39. uchar *p = NULL;
  40. for (int x = ; x<img->height;++x)
  41. {
  42. p = (uchar*)(img->imageData + img->widthStep*x);
  43. for (int y = ; y<img->width;++y)
  44. {
  45. p[y] = T[p[y]];
  46. }
  47. }
  48. }
  49.  
  50. // 生成高斯分布
  51. void GenerateGaussModel(double model[])
  52. {
  53. double m1,m2,sigma1,sigma2,A1,A2,K;
  54. m1 = 0.15;
  55. m2 = 0.75;
  56. sigma1 = 0.05;
  57. sigma2 = 0.05;
  58. A1 = ;
  59. A2 = 0.07;
  60. K = 0.002;
  61.  
  62. double c1 = A1*(1.0/(sqrt(*CV_PI))*sigma1);
  63. double k1 = *sigma1*sigma1;
  64. double c2 = A2*(1.0/(sqrt(*CV_PI))*sigma2);
  65. double k2 = *sigma2*sigma2;
  66. double p = 0.0,val= 0.0,z = 0.0;
  67. for (int zt = ;zt < ;++zt)
  68. {
  69. val = K + c1*exp(-(z-m1)*(z-m1)/k1) + c2*exp(-(z-m2)*(z-m2)/k2);
  70. model[zt] = val;
  71. p = p +val;
  72. z = z + 1.0/;
  73. }
  74. for (int i = ;i<; ++i)
  75. {
  76. model[i] = model[i]/p;
  77. }
  78. }

cvCalcHist

将图像规定化为高斯分布函数。

算法演示:

计算变换函数 根据 规定的直方图 得来。但非严格单调,所以需要:

    (a:s值) --> (b:z值)    

发现:S0G(Z3)竟然一致,正好匹配。

对比直方图

通过直方图匹配,可以使图像的相似度变高,但也仅仅是从颜色的角度。

  1. int main()
  2. {
  3. IplImage * image= cvLoadImage("myhand1.jpg");
  4. IplImage * image2= cvLoadImage("myhand2.jpg");
  5. int hist_size=;
  6. float range[] = {,};
  7. float* ranges[]={range};
  8.  
  9. IplImage* gray_plane = cvCreateImage(cvGetSize(image),,);
  10. cvCvtColor(image,gray_plane,CV_BGR2GRAY);
  11. CvHistogram* gray_hist = cvCreateHist(,&hist_size,CV_HIST_ARRAY,ranges,);
  12. cvCalcHist(&gray_plane,gray_hist,,);
  13.  
  14. IplImage* gray_plane2 = cvCreateImage(cvGetSize(image2),,);
  15. cvCvtColor(image2,gray_plane2,CV_BGR2GRAY);
  16. CvHistogram* gray_hist2 = cvCreateHist(,&hist_size,CV_HIST_ARRAY,ranges,);
  17. cvCalcHist(&gray_plane2,gray_hist2,,);
  18.  
  19. //相关:CV_COMP_CORREL
  20. //卡方:CV_COMP_CHISQR
  21. //直方图相交:CV_COMP_INTERSECT
  22. //Bhattacharyya距离:CV_COMP_BHATTACHARYYA
  23. double com=cvCompareHist(gray_hist,gray_hist2,CV_COMP_BHATTACHARYYA);
  24.  
  25. cout<<com<<endl;
  26. }

cvCompareHist

cvCompareHist的结果为【0,1】的浮点数,越小表示两幅图匹配度越高,0.0时两幅图精确匹配。  

提供的对比方法有四种:

     

局部增强 (涉及到统计,大有可为)

分辨亮区域与暗区域的不同,同时只增强暗区域。

[OpenCV] Image Processing - Grayscale Transform & Histogram的更多相关文章

  1. [OpenCV] Image Processing - Grayscale Transform

    "每个像素的输出值只取决于其输入值" 重难点:Histogram equalization 参考:笑得很甜 http://blog.csdn.net/xiaowei_cqu/art ...

  2. [OpenCV] Image Processing - Fuzzy Set

    使用模糊技术进行 (灰度变换Grayscale Transform) 和 (空间滤波Spatial Filtering) 模糊集合为处理不严密信息提供了一种形式. 首先,需要将输入量折算为隶属度,这个 ...

  3. 使用OpenMP加快OpenCV图像处理性能 | speed up opencv image processing with openmp

    本文首发于个人博客https://kezunlin.me/post/7a6ba82e/,欢迎阅读! speed up opencv image processing with openmp Serie ...

  4. 使用OpenCL提升OpenCV图像处理性能 | speed up opencv image processing with OpenCL

    本文首发于个人博客https://kezunlin.me/post/59afd8b3/,欢迎阅读最新内容! speed up opencv image processing with OpenCL G ...

  5. opencv python:图像直方图 histogram

    直接用matplotlib画出直方图 def plot_demo(image): plt.hist(image.ravel(), 256, [0, 256]) # image.ravel()将图像展开 ...

  6. [OpenCV] Image Processing - Spatial Filtering

    "利用给定像素周围的像素的值决定此像素的最终的输出值“ 教学效果: 策略: 1. 拉普拉斯,突出小细节: . 梯度,突出边缘: . 平滑过的梯度图像用于掩蔽: . 灰度变换,增加灰度动态范围 ...

  7. [OpenCV] Image Processing - Image Elementary Knowledge

    "没有坚实的理论基础,实践只会浅尝于表面." 这是两本打基础的书,没系统学过的话,怎么好意思说自己会CV. 该领域,兴军亮 这个名字屡次出现,看来是计算机视觉领域国内的年轻才俊,向 ...

  8. [Python] python vs cplusplus

    一些学习过程中的总结的两种语言的小对比,帮助理解OO programming. Continue... 字典 序列 --> 字典 Python: def get_counts(sequence) ...

  9. 本人AI知识体系导航 - AI menu

    Relevant Readable Links Name Interesting topic Comment Edwin Chen 非参贝叶斯   徐亦达老板 Dirichlet Process 学习 ...

随机推荐

  1. java随机范围内的日期

    使用了最新的java8的java.time类,并提供了LocalDateTime和java.util.Date之间的转换. 使用方法: randomLocalDateTime(-3,3) : 取距离今 ...

  2. [sed] linux sed 批量替换字符串

    比如,要将目录/modules下面所有文件中的zhangsan都修改成lisi,这样做: sed -i "s/zhangsan/lisi/g" `grep zhangsan -rl ...

  3. linq to sql之组装where条件下的'或'语句

    之前遇到过类似的需求,即前台传入几个过滤条件,后台动态组装where. 例如,前台传入name='张三',age=10, 其余的字段,类似email,QQ之类的本次查询时不做过滤. 用linq to ...

  4. android开发(38) 使用 DrawerLayou t实现左侧抽屉式导航菜单

    最近流行 左侧抽屉式的导航条菜单,知乎,360,QQ都使用了这样的导航菜单,我们也了解下: Android Design 的流行趋势:Navigation Drawer 导航抽屉 参考这篇文章:htt ...

  5. C#内置泛型委托:Func委托

    1.什么是Func委托 Func委托代表有返回类型的委托 2.Func委托定义 查看Func的定义: using System.Runtime.CompilerServices; namespace ...

  6. 删除json对象中空值

    function deleteEmptyProperty(obj){ var object = obj; for (var i in object) { var value = object[i]; ...

  7. 深入浅出学习Hibernate框架(二):JDBC基础操作

    上篇博客<深入浅出学习Hibernate框架(一):从实例入手初识Hibernate框架>简单介绍了一下Hibernate框架,并且举了一个实例来了解Hibernate.这篇博客将介绍JD ...

  8. Sword protobuf学习三

    #include <iostream> #include <sys/types.h> /* See NOTES */ #include <sys/socket.h> ...

  9. Qt之QLocalSocket

    简述 QLocalSocket类提供了一个本地socket. 在Windows中,这是一个命名管道:在Unix中,这是一个本地网域socket. 如果发生错误,socketError()会返回错误的类 ...

  10. Java HashSet和HashMap源码剖析

    转自: Java HashSet和HashMap源码剖析 总体介绍 之所以把HashSet和HashMap放在一起讲解,是因为二者在Java里有着相同的实现,前者仅仅是对后者做了一层包装,也就是说Ha ...