直方图

转载自http://blog.csdn.net/lu597203933/article/details/14104505


灰度直方图

这次我从最基本的直方图讲起,一维直方图,至于二维等高维直方图,仅作为了解,后面有时间另开blog详解。

1:首先我们给出一幅图


其中的数据假设对应一副灰度图片的灰度值,则直方图的作用就是画出在bin范围内取值的个数,如图左边的直方图---相当于我们word中的柱状图。

2:在opencv中直方图的结构为:

typedef struct CvHistogram

{

   int     type;    // 不用管

   CvArr*  bins;存放直方图在每一维上直方柱的具体数据,由于存在多维直方图。如果是一维直方图,那么bins就是一个一维的矩阵;如果是二维直方图,那么bins就是一个二维的矩阵,等等。

   float   thresh[CV_MAX_DIM][2]; /* 直方柱的划分 是统一划分的,即均等划分的 */

   float** thresh2; /*不均等划分,可以自动设定每一个直方柱的取值范围。之所以是二级指针,每一个柱的取值范围用一级指针表示,又存在多个直方柱,所以需要二级指针才能表示。每一个不在指定范围的值,会被忽略掉。*/

    CvMatND mat; /* 存放直方图的数据 */

}

3:直方图的创建:

	CvHistogram* cvCreateHist( int dims, int* sizes, int type, float** ranges=NULL, int uniform=1);
	其中: 	

dims:表示直方图的维度

sizes:每一维上直方柱(bin)的数据 ----- 创建多大的矩阵  如一维则size[0] =256

二维:则size[0]=256,size[1] = 256;;

type  ---- 创建一个什么样的矩阵

直方图存储数据的方式:

CV_HIST_ARRAY意味着直方图数据表示为多维密集数组CvMatND;

         CV_HIST_TREE 意味着直方图数据表示为多维稀疏数组CvSparseMat.---- 大于0才开设相应的空间存储它<高维的必选它>

Ranges:

直方图在每一维度上的范围。

         如果是一维则代码为:

                                               Floatrange[] = {0,255};

                                               Float*ranges[]={range};

         如果是二维则代码为:

                                               Floatb_range[] = {0,255};

                                               Floatg_range[] ={0,255};

                                               Float*ranges[] = {b_range, g_range};

uniform

该值为0时,表示bin的范围是程序员自由设定的。当该值为非零时,表示bin的划分,是均等划分。

4:创建一个直方图后,避免里面存在一些随机值,我们可以将其清除cvClearHist

代码: cvClearHist(hist);

5:计算图像的直方图  cvCalcHist

         代码:cvCalcHist(&imgBlue, hist, 0, 0);

6:将得到的直方图画出来: 其中用到了函数 cvGetMinMaxHistVaule, cvQueryHistValue_1D, cvFillConvexPoly.

[cpp] view plaincopyprint?
  1. IplImage *DrawHistogram(CvHistogram*hist, float scaleX = 1, float scaleY = 1){  // 画直方图  
  2.     float histMax = 0;  
  3.     cvGetMinMaxHistValue(hist, 0 , &histMax, 0, 0);  // 取得直方图中的最值  
  4.     IplImage *imgHist = cvCreateImage(cvSize(256 * scaleX, 64*scaleY), 8, 1);  
  5.     cvZero(imgHist);  清空随机值  
  6.     for(int i = 0; i < 255; i++)  
  7.     {  
  8.         float histValue = cvQueryHistValue_1D(hist, i); // 取得直方图中的i值  
  9.         float nextValue = cvQueryHistValue_1D(hist, i+1);  
  10.         int numPt = 5;  
  11.         CvPoint pt[5];  
  12.         pt[0] = cvPoint(i*scaleX, 64*scaleY);  
  13.         pt[1] = cvPoint((i+1)*scaleX, 64*scaleY);  
  14.         pt[2] = cvPoint((i+1)*scaleX, (1 -(nextValue/histMax))* 64 * scaleY);  
  15.         pt[3] = cvPoint((i+1)*scaleX, (1 -(histValue/histMax))* 64 * scaleY);  
  16.         pt[4] = cvPoint(i*scaleX, 64*scaleY);  
  17.         cvFillConvexPoly(imgHist, pt, numPt, cvScalarAll(255));  
  18.     }  
  19.     return imgHist;  
  20.   
  21. }  
总结:以上给出得到直方图的主要步骤:直方图的创建—>计算图像的直方图—>画出直方图。以下给出完整的代码及运行的结果

[cpp] view plaincopyprint?
  1. #include <iostream>  
  2. #include "cv.h"  
  3. #include "highgui.h"  
  4. #include "cxcore.h"  
  5. using namespace std;  
  6.   
  7. IplImage *DrawHistogram(CvHistogram*hist, float scaleX = 1, float scaleY = 1){  // 画直方图  
  8.     float histMax = 0;  
  9.     cvGetMinMaxHistValue(hist, 0 , &histMax, 0, 0);  // 取得直方图中的最值  
  10.     IplImage *imgHist = cvCreateImage(cvSize(256 * scaleX, 64*scaleY), 8, 1);  
  11.     cvZero(imgHist);  清空随机值  
  12.     for(int i = 0; i < 255; i++)  
  13.     {  
  14.         float histValue = cvQueryHistValue_1D(hist, i); // 取得直方图中的i值  
  15.         float nextValue = cvQueryHistValue_1D(hist, i+1);  
  16.         int numPt = 5;  
  17.         CvPoint pt[5];  
  18.         pt[0] = cvPoint(i*scaleX, 64*scaleY);  
  19.         pt[1] = cvPoint((i+1)*scaleX, 64*scaleY);  
  20.         pt[2] = cvPoint((i+1)*scaleX, (1 -(nextValue/histMax))* 64 * scaleY);  
  21.         pt[3] = cvPoint((i+1)*scaleX, (1 -(histValue/histMax))* 64 * scaleY);  
  22.         pt[4] = cvPoint(i*scaleX, 64*scaleY);  
  23.         cvFillConvexPoly(imgHist, pt, numPt, cvScalarAll(255));  
  24.     }  
  25.     return imgHist;  
  26.   
  27.   
  28. }  
  29.   
  30.   
  31. int main()  
  32. {  
  33.     IplImage *img = cvLoadImage("F:\\tongtong.jpg",1);  
  34.     if(!img){  
  35.         cout << "No data img" << endl;  
  36.     }  
  37.     int dims = 1;  
  38.     int sizes = 256;  
  39.     float range[] = {0,255};  
  40.     float*ranges[]={range};  
  41.     CvHistogram *hist = cvCreateHist(dims, &sizes, CV_HIST_ARRAY, ranges, 1);  
  42.     cvClearHist(hist);          //清除直方图里面的随机值   
  43.     IplImage *imgBlue = cvCreateImage(cvGetSize(img), 8, 1);  
  44.     IplImage *imgGreen = cvCreateImage(cvGetSize(img), 8, 1);  
  45.     IplImage *imgRed = cvCreateImage(cvGetSize(img), 8, 1);  
  46.     cvSplit(img, imgBlue, imgGreen, imgRed, NULL);   //将多通道图像分解  
  47.   
  48.     cvCalcHist(&imgBlue, hist, 0, 0);    // 计算图像的直方图  
  49.     IplImage *histBlue = DrawHistogram(hist);   // 将直方图中的数据画出来  
  50.     cvClearHist(hist);  
  51.   
  52.     cvCalcHist(&imgGreen, hist, 0, 0);    
  53.     IplImage *histGreen = DrawHistogram(hist);  
  54.     cvClearHist(hist);  
  55.     cvCalcHist(&imgRed, hist, 0, 0);  
  56.     IplImage *histRed = DrawHistogram(hist);  
  57.     cvClearHist(hist);  
  58.   
  59.     cvNamedWindow("show",0);  
  60.     cvNamedWindow("B", 0);  
  61.     cvNamedWindow("G", 0);  
  62.     cvNamedWindow("R", 0);  
  63.     cvShowImage("show",img);  
  64.     cvShowImage("B",histBlue);  
  65.     cvShowImage("G",histGreen);  
  66.     cvShowImage("R", histRed);  
  67.     cvWaitKey(0);  
  68.     cvReleaseImage(&img);  
  69.     cvDestroyWindow("show");  
  70.     cvReleaseImage(&histBlue);  
  71.     cvDestroyWindow("B");  
  72.     cvReleaseImage(&histGreen);  
  73.     cvDestroyWindow("G");  
  74.     cvReleaseImage(&histRed);  
  75.     cvDestroyWindow("R");  
  76.     return 0;  
  77.       
  78. }  
运行结果:

/*这部分是后面添加的,没有另开blog进行讲解了,其实道理一样的*/

下面简单介绍二维的情况:要统计一个二维直方图,二维坐标+统计值,这就是三维的结构,就会变成一个3D图像,其实不然。我们的处理技巧是将二维坐标分别作为x和y轴,而统计值作为颜色值。如代码cvRectangle(imgHist, cvPoint(i*scaleX, j*scaleY), cvPoint((i+1)*scaleX- 1, (j+1)*scaleY - 1), CV_RGB(intensity,intensity,intensity), CV_FILLED); i和j分别代表坐标,intensity即为统计值。

Code:

[cpp] view plaincopyprint?
  1. IplImage *DrawHistogram2(CvHistogram*hist, float scaleX = 1, float scaleY = 1){  // 画直方图  
  2.     float histMax = 0;  
  3.     cvGetMinMaxHistValue(hist, 0 , &histMax, 0, 0);  // 取得直方图中的最值  
  4.     IplImage *imgHist = cvCreateImage(cvSize(256 * scaleX, 256*scaleY), 8, 3);  
  5.     cvZero(imgHist);  清空随机值  
  6.     for(int i = 0; i < 255; i++)  
  7.     {  
  8.         for(int j = 0; j < 255; j++)  
  9.         {  
  10.             float histValue = cvQueryHistValue_2D(hist, i, j); // 取得直方图中的i值  
  11.             int intensity = cvRound(histValue * 255 /histMax);  
  12.             float nextValue = cvQueryHistValue_1D(hist, i+1);  
  13.             int numPt = 5;  
  14.             CvPoint pt[5];  
  15.             pt[0] = cvPoint(i*scaleX, j*scaleY);  
  16.             pt[1] = cvPoint(i*scaleX, (j+1)*scaleY - 1);  
  17.             pt[2] = cvPoint((i+1)*scaleX - 1, (j+1)*scaleY - 1);  
  18.             pt[3] = cvPoint((i+1)*scaleX - 1, j*scaleY);  
  19.             pt[4] = cvPoint(i*scaleX, j*scaleY);  
  20.             cvFillConvexPoly(imgHist, pt, numPt, cvScalarAll(intensity));  
  21.             //cvRectangle(imgHist, cvPoint(i*scaleX, j*scaleY), cvPoint((i+1)*scaleX - 1, (j+1)*scaleY - 1), CV_RGB(intensity,intensity,intensity), CV_FILLED);  
  22.             // (i+1)*scaleX - 1代表下一个坐标点(i+1)*scaleX的最后一点  
  23.         }  
  24.     }  
  25.     return imgHist;  
  26. }  
  27.   
  28. int main()  
  29. {  
  30.     IplImage *img = cvLoadImage("F:\\baboon.jpg",1);  
  31.     if(!img){  
  32.         cout << "No data img" << endl;  
  33.     }  
  34.     int dims = 2;  
  35.     int sizes[2] = {256, 256};  // 创建多大的矩阵  
  36.     float b_range[] = {0,255};    // 矩阵的取值范围  
  37.     float g_range[] = {0, 255};  
  38.     float*ranges[]={b_range, g_range};  
  39.     CvHistogram *hist = cvCreateHist(dims, sizes, CV_HIST_ARRAY, ranges, 1);  
  40.     cvClearHist(hist);          //清除直方图里面的随机值   
  41.     IplImage *imgBlue = cvCreateImage(cvGetSize(img), 8, 1);  
  42.     IplImage *imgGreen = cvCreateImage(cvGetSize(img), 8, 1);  
  43.     IplImage *imgRed = cvCreateImage(cvGetSize(img), 8, 1);  
  44.     IplImage *planes[] = {imgBlue, imgGreen};  // 二维是这样处理的********  
  45.     cvSplit(img, imgBlue, imgGreen, imgRed, NULL);   //将多通道图像分解  
  46.       
  47.     cvCalcHist(planes,hist, 0, 0);  
  48.     IplImage *bg_image =  DrawHistogram2(hist);  
  49.     cvNamedWindow("show",0);  
  50.     cvShowImage("show",img);  
  51.     cvNamedWindow("bg_image");  
  52.     cvShowImage("bg_image",bg_image);  
  53.     cvWaitKey(0);  
  54.     cvReleaseImage(&img);  
  55.     cvReleaseImage(&bg_image);  
  56.     cvDestroyWindow("show");  
  57.     cvDestroyWindow("bg_image");  
  58.     return 0;  
  59.       
  60. }  

Result:

作者:小村长  出处:http://blog.csdn.net/lu597203933 欢迎转载或分享,但请务必声明文章出处。 (新浪微博:小村长zack, 欢迎交流!)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/409182.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

自己寫的 Loading JS插件

本文為原創文章&#xff0c;轉載請注明出處&#xff0c;謝謝。 1 /**2 * author samkin.yang3 * version 1.04 */5 var $_yxj new SamkinLoading(); 6 7 8 (function($){9 if($){ 10 $.fn.extend({ 11 showLoading : function(){ 12…

如何设置多级标题_办公技巧|标题样式amp;多级列表结合,再也不用手动修改1.1、1.2 ...!...

每周一句Accept the things you cannot change. Have the courage to change the things you can.接受那些你也无法改变的事&#xff0c;而能够改变的事则要勇于行动。1. 标题样式&多级列表结合编制方案、修改合同时&#xff0c;如果需要增加或者删除某一章节、条款&#x…

包r语言_R语言交互式可视化包CanvasXpress

CanvasXpress包简介CanvasXpress核心是一个JavaScript语言编写的库&#xff0c;主要用于可重复研究。在R中可以安装canvasXpress包&#xff0c;使用canvasXpress命令绘制各种各样的交互式图形&#xff0c;非常高效方便&#xff0c;实现如下数百种图形&#xff0c;令你的结果瞬间…

widget(6、menu)

menu是android菜单栏控件&#xff0c;选择手机menu键时触发&#xff0c;基本函数如下&#xff1a; public boolean onCreateOptionsMenu(Menu menu) {MenuItem item2 menu.add(Menu.NONE, MENU_ID2, Menu.NONE, "菜单项2");MenuItem item3 menu.add(Menu.NONE, MEN…

用函数计算工龄_用Excel计算财务账期,离不开这3个函数

正文共&#xff1a;1577 字 6 图预计阅读时间&#xff1a; 4 分钟在很多企业&#xff0c;应收账款要按指定的账期显示&#xff0c;类似“0-30天&#xff0c;31-60天&#xff0c;……”这种样式。在Excel应收账款分析表中&#xff0c;需要根据应收账款的账龄天数显示不同的账期区…

android所有颜色代码

转载&#xff1a;http://www.cnblogs.com/elleniou/archive/2012/04/25/2469676.html <?xml version"1.0" encoding"utf-8" ?> <resources> <color name"white">#ffffff</color><!--白色 --> <color nam…

排序算法--冒泡排序的首尾改进

在排序算法中&#xff0c;冒泡排序是一个很经典的算法&#xff0c;最初的冒泡排序一直要运行n-1次&#xff0c;但是其中有些事不必要的操作&#xff0c;例 如&#xff0c;当没有两个数据发生交换时&#xff0c;就可以结束运行。 本文介绍的一种方法是对上述条件的改进&#xff…

Java学习笔记之基础应用(2015.3.21)

最近打算好好学习下Java&#xff0c;所以打算记录下一些知识点或者是值得记住需要记住的代码吧&#xff01; 1.灵活使用位运算实现加密算法 可以使用“^”异或运算符把字符串与一个特定值进行异或运算&#xff0c;这样就可以得到一个加密后的字符串&#xff0c;这就是一个简单…

dw网页设计期末设计一个网页_Dreamweaver网页设计期末模拟试题(1)

山东广播电视大学开放教育Dreamweaver网页设计期末模拟试题 (1)一、单项选择题1&#xff0e;下图为Dreamweaver 8的新建文档页面&#xff0c;一般情况下&#xff0c;创建完全空白的静态页面应选择()。A. 基本页类别中的“HTML”选项B. 基本页类别中的“HTML模板”选项C. 动态页…

直接拿来用!10款实用Android UI工具

转载地址&#xff1a;http://blog.csdn.net/bboyfeiyu/article/details/13295233 移动应用的UI设计就好似达摩克利斯之剑&#xff0c;一方面&#xff0c;一个视觉、交互、体验良好的UI可以加强应用在用户心目中的形象和识别性。而另一方面&#xff0c;一个体验糟糕的UI设计不仅…

o_rdonly_O_RDWR, O_CREAT等open函数标志位在哪里定义? | 学步园

查了下O_RDWR, O_CREAT等定义&#xff0c;终于找到了。我的系统是Fedora12, 其定义在文件&#xff1a;/usr/include/asm-generic/fcntl.h&#xff0c;部分定义如下&#xff1a;#include /* open/fcntl - O_SYNC is only implemented on blocks devices and on fileslocated on …

上百个Android开源项目分享

转载地址&#xff1a;[http://blog.csdn.net/bboyfeiyu/article/details/12234163] 上百个Android开源项目分享&#xff0c;希望对android开发有帮助。 Android PDF 阅读器 http://sourceforge.net/projects/andpdf/files/ 个人记账工具 OnMyMeans http://sourceforge.net/p…

批量读入一个文件夹中文件的数据操作实例

批量处理 > coo dir("test") > path "/panfs/TC_FUN/USER/group3/yanzengli/other/study/R_study/knowledge/test" > doc_path sapply(coo, function(names) paste(path, names, sep/)) >doc <- sapply( doc_path, function(doc) readLi…