Opencv回顾--直方图统计和直方图均衡化

主要函数
直方图计算

void calcHist(const Mat* images, int nimages, const int* channels, InputArray mask, OutputArray hist, int dims, const int* histSize, const float** ranges, bool uniform=true, bool accumulate=false )

onst Mat* images:输入图像

 int nimages:输入图像的个数

const int* channels:需要统计直方图的第几通道

InputArray mask:掩膜,,计算掩膜内的直方图  ...Mat()

OutputArray hist:输出的直方图数组

int dims:需要统计直方图通道的个数

const int* histSize:指的是直方图分成多少个区间,就是 bin的个数

const float** ranges: 统计像素值得区间

bool uniform=true::是否对得到的直方图数组进行归一化处理

bool accumulate=false:在多个图像时,是否累计计算像素值得个数

归一化数据。该函数分为范围归一化与数据值归一化

void cv::normalize(InputArry src,InputOutputArray dst,double alpha=1,double beta=0,int norm_type=NORM_L2,int dtype=-1,InputArray mark=noArry())
src               输入数组;
dst               输出数组,数组的大小和原数组一致;
alpha           1,用来规范值,2.规范范围,并且是下限;
beta             只用来规范范围并且是上限;
norm_type   归一化选择的数学公式类型;
dtype           当为负,输出在大小深度通道数都等于输入,当为正,输出只在深度与输如不同,不同的地方游dtype决定;
mark            掩码。选择感兴趣区域,选定后只能对该区域进行操作。

几种归一化方式

 vector positiveData = { 2.0, 8.0, 10.0 }; 输入的值
    vector normalizedData_l1, normalizedData_l2, normalizedData_inf, normalizedData_minmax;

    // Norm to probability (total count)
    // sum(numbers) = 20.0
    // 2.0      0.1     (2.0/20.0)
    // 8.0      0.4     (8.0/20.0)
    // 10.0     0.5     (10.0/20.0)
    normalize(positiveData, normalizedData_l1, 1.0, 0.0, NORM_L1);

    // Norm to unit vector: ||positiveData|| = 1.0
    // 2.0      0.15
    // 8.0      0.62
    // 10.0     0.77
    normalize(positiveData, normalizedData_l2, 1.0, 0.0, NORM_L2);

    // Norm to max element
    // 2.0      0.2     (2.0/10.0)
    // 8.0      0.8     (8.0/10.0)
    // 10.0     1.0     (10.0/10.0)
    normalize(positiveData, normalizedData_inf, 1.0, 0.0, NORM_INF);

    // Norm to range [0.0;1.0]
    // 2.0      0.0     (shift to left border)
    // 8.0      0.75    (6.0/8.0)
    // 10.0     1.0     (shift to right border)
    normalize(positiveData, normalizedData_minmax, 1.0, 0.0, NORM_MINMAX);

写的案例

using namespace cv;
using namespace std;
const int bins = 256;
Mat src;
const char* winTitle = "input image";
void showHistogram();
int main(int argc, char** argv) {
	src = imread("12345.jpg");
	if (src.empty()) {
		printf("could not load image...\n");
		return 0;
	}
	namedWindow(winTitle, WINDOW_AUTOSIZE);
	imshow(winTitle, src);
	showHistogram();
	waitKey(0);
	return 0;
}

void showHistogram() {
	// 三通道分离
	vector bgr_plane;
	split(src, bgr_plane);
	// 定义参数变量
	const int bins[1] = { 256 };
	float hranges[2] = { 0,255 };
	const float* ranges[1] = { hranges };
	Mat b_hist;
	Mat g_hist;
	Mat r_hist;
	// 计算Blue, Green, Red通道的直方图
	calcHist(&bgr_plane[0], 1, 0, Mat(), b_hist, 1, bins, ranges);
	//CV_EXPORTS void calcHist(
	//	const Mat* images,      输入图像用于计算的图像
	//	int nimages,       	    输入的图像个数 
	//	const int* channels,    需要统计直方图的第几通道
	//	InputArray mask,        掩膜,,计算掩膜内的直方图  ...Mat()
	//	OutputArray hist,       输出的直方图数组
	//	int dims,               需要统计直方图通道的个数
	//	const int* histSize,    指的是直方图分成多少个区间,就是 bin的个数
	//	const float** ranges,   统计像素值得区间
	//	bool uniform = true,    是否对得到的直方图数组进行归一化处理
	//	bool accumulate = false 在多个图像时,是否累计计算像素值得个数
	//);
	//ranges:用来进行统计的范围。比如 float rang1[] = {0, 20};float rang2[] = {30, 40};
	//const float*rangs[] = {rang1, rang2};那么就是对0,20和30,40范围的值进行统计。
	calcHist(&bgr_plane[1], 1, 0, Mat(), g_hist, 1, bins, ranges);
	calcHist(&bgr_plane[2], 1, 0, Mat(), r_hist, 1, bins, ranges);
	// 显示直方图
	int hist_w = 512;
	//展示用图的宽
	int hist_h = 400;
	//展示用图的高
	int bin_w = cvRound((double)hist_w / bins[0]);
	//单位距离 规整到256个点的基准单位
	Mat histImage = Mat::zeros(hist_h, hist_w, CV_8UC3);
	
	//建立一个展示用图
	// 归一化直方图数据
	normalize(b_hist, b_hist, 0, histImage.rows, NORM_MINMAX, -1, Mat());
	//归一化整体数据将统计的数值规整到0到400
	normalize(g_hist, g_hist, 0, histImage.rows, NORM_MINMAX, -1, Mat());
	normalize(r_hist, r_hist, 0, histImage.rows, NORM_MINMAX, -1, Mat());
	// 绘制直方图曲线
	//开始循环 一共256个bin 从第一个点到第二个点做一条线 继续从第n个点到n+1个点画线
	for (int i = 1; i < bins[0]; i++) {
		line(histImage, Point(bin_w*(i - 1), hist_h - cvRound(b_hist.at(i - 1))),
			Point(bin_w*(i), hist_h - cvRound(b_hist.at(i))), Scalar(255, 0, 0), 2, 8, 0);
		//第一个点 X坐标 单位距离1*bin_w Y坐标 图像高减去当前bin归一化后的值
		//第一个点 X坐标 单位距离2*bin_w Y坐标 图像高减去当前归一化后的值
		line(histImage, Point(bin_w*(i - 1), hist_h - cvRound(g_hist.at(i - 1))),
			Point(bin_w*(i), hist_h - cvRound(g_hist.at(i))), Scalar(0, 255, 0), 2, 8, 0);
		line(histImage, Point(bin_w*(i - 1), hist_h - cvRound(r_hist.at(i - 1))),
			Point(bin_w*(i), hist_h - cvRound(r_hist.at(i))), Scalar(0, 0, 255), 2, 8, 0);
	}
	// 显示直方图
	namedWindow("Histogram Demo", WINDOW_AUTOSIZE);
	imshow("Histogram Demo", histImage);
}

Opencv回顾--直方图统计和直方图均衡化_第1张图片
均衡化直方图
函数功能:直方图均衡化,该函数能归一化图像亮度和增强对比度用来提升图像质量
直方图均衡化是怎样做到的?
均衡化指的是把一个分布 (给定的直方图) 映射 到另一个分布 (一个更宽更统一的强度值分布), 所以强度值分布会在整个范围内展开.

void equalizeHist( InputArray src, OutputArray dst );
src 8位单通道图像。
dst与src具有相同大小和类型的目标图像。

图像直方图比较
图像直方图比较,就是计算两幅图像的直方图数据,比较两组数据的相似性,从而得到两幅图像之间的相似程度,直方图比较在早期的CBIR中是应用很常见的技术手段,通常会结合边缘处理、词袋等技术一起使用。
OpenCV中直方图比较的API很简单

compareHist(hist1, hist2, method)

CV_EXPORTS_W double compareHist( InputArray H1, InputArray H2, int method );

method 值可以是HistCompMethods

enum HistCompMethods {
	相关性比较(Correlation)
    	HISTCMP_CORREL=0,
	Chi-Square(卡方比较)
	    HISTCMP_CHISQR=1,
	Intersection(十字交叉性)
 	   HISTCMP_INTERSECT=2,
	Bhattacharyya distance(巴氏距离)
   	 HISTCMP_BHATTACHARYYA=3,
	Alternative Chi-Square (代替卡方 ?这个不清楚)
  	  HISTCMP_CHISQR_ALT=4,
	Kullback-Leibler divergence(Kullback-Leibler散度)
  	  HISTCMP_KL_DIV=5
  	  }

你可能感兴趣的:(C/C++复习回顾,Opencv)