亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

C++?OpenCV實現(xiàn)白平衡之完美反射算法

 更新時間:2022年05月30日 09:58:30   作者:翟天保Steven  
完美反射算法是白平衡各種算法中較常見的一種,比灰度世界算法更優(yōu)。本文將利用C++和OpenCV實現(xiàn)白平衡中的完美反射算法,需要的可以參考一下

實現(xiàn)原理

白平衡的意義在于,對在特定光源下拍攝時出現(xiàn)的偏色現(xiàn)象,通過加強對應(yīng)的補色來進(jìn)行補償,使白色物體能還原為白色。

完美反射算法是白平衡各種算法中較常見的一種,比灰度世界算法更優(yōu)。它假設(shè)圖像世界中最亮的白點是一個鏡面,能完美反射光照;基于白點,將三通道的數(shù)值進(jìn)行適當(dāng)?shù)卣{(diào)整,以達(dá)到白平衡效果;除此之外,還需要統(tǒng)計最亮的一定區(qū)間的三通道均值,該均值與該通道最大值的差距決定了該通道調(diào)整的力度。

通俗的講,若圖像中綠色分量最大值是255,但是綠色最亮的前百分之10個點的平均值只有80,說明原圖的綠色分量整體較低,需要對其加強;若最大值只有100,那么加強的系數(shù)就較低,白平衡的效果就不達(dá)預(yù)期。這就是完美反射算法比較依賴圖像中存在白點的原因,白點的三通道灰度值接近【255,255,255】。最下方將用實際圖像作進(jìn)一步說明,以幫助讀者理解。

完美反射算法的實現(xiàn)流程如下:   

1.計算圖像RGB三通道各自的灰度最大值Rmax、Gmax、Bmax。

2.利用三通道數(shù)值和,確定圖像最亮區(qū)間的下限T。

3.計算圖像三通道數(shù)值和大于T的點的三通道均值Rm、Gm、Bm。

4.計算三通道的補償系數(shù),即單通道最大值除以單通道亮區(qū)平均值。

功能函數(shù)代碼

// 白平衡-完美反射
cv::Mat WhiteBalcane_PRA(cv::Mat src)
{
	cv::Mat result = src.clone();
	if (src.channels() != 3)
	{
		cout << "The number of image channels is not 3." << endl;
		return result;
	}
 
	// 通道分離
	vector<cv::Mat> Channel;
	cv::split(src, Channel);
 
	// 定義參數(shù)
	int row = src.rows;
	int col = src.cols;
	int RGBSum[766] = { 0 };
	uchar maxR, maxG, maxB;
 
	// 計算單通道最大值
	for (int i = 0; i < row; ++i)
	{
		uchar *b = Channel[0].ptr<uchar>(i);
		uchar *g = Channel[1].ptr<uchar>(i);
		uchar *r = Channel[2].ptr<uchar>(i);
		for (int j = 0; j < col; ++j)
		{
			int sum = b[j] + g[j] + r[j];
			RGBSum[sum]++;
			maxB = max(maxB, b[j]);
			maxG = max(maxG, g[j]);
			maxR = max(maxR, r[j]);
		}
	}
 
	// 計算最亮區(qū)間下限T
	int T = 0;
	int num = 0;
	int K = static_cast<int>(row * col * 0.1);
	for (int i = 765; i >= 0; --i)
	{
		num += RGBSum[i];
		if (num > K)
		{
			T = i;
			break;
		}
	}
	
	// 計算單通道亮區(qū)平均值
	double Bm = 0.0, Gm = 0.0, Rm = 0.0;
	int count = 0;
	for (int i = 0; i < row; ++i)
	{
		uchar *b = Channel[0].ptr<uchar>(i);
		uchar *g = Channel[1].ptr<uchar>(i);
		uchar *r = Channel[2].ptr<uchar>(i);
		for (int j = 0; j < col; ++j)
		{
			int sum = b[j] + g[j] + r[j];
			if (sum > T)
			{
				Bm += b[j];
				Gm += g[j];
				Rm += r[j];
				count++;
			}
 
		}
	}
	Bm /= count;
	Gm /= count;
	Rm /= count;
 
	// 通道調(diào)整
	Channel[0] *= maxB / Bm;
	Channel[1] *= maxG / Gm;
	Channel[2] *= maxR / Rm;
 
	// 合并通道
	cv::merge(Channel, result);
 
	return result;
}

C++測試代碼

#include <iostream>
#include <opencv.hpp>
 
using namespace std;
 
// 白平衡-完美反射
cv::Mat WhiteBalcane_PRA(cv::Mat src)
{
	cv::Mat result = src.clone();
	if (src.channels() != 3)
	{
		cout << "The number of image channels is not 3." << endl;
		return result;
	}
 
	// 通道分離
	vector<cv::Mat> Channel;
	cv::split(src, Channel);
 
	// 定義參數(shù)
	int row = src.rows;
	int col = src.cols;
	int RGBSum[766] = { 0 };
	uchar maxR, maxG, maxB;
 
	// 計算單通道最大值
	for (int i = 0; i < row; ++i)
	{
		uchar *b = Channel[0].ptr<uchar>(i);
		uchar *g = Channel[1].ptr<uchar>(i);
		uchar *r = Channel[2].ptr<uchar>(i);
		for (int j = 0; j < col; ++j)
		{
			int sum = b[j] + g[j] + r[j];
			RGBSum[sum]++;
			maxB = max(maxB, b[j]);
			maxG = max(maxG, g[j]);
			maxR = max(maxR, r[j]);
		}
	}
 
	// 計算最亮區(qū)間下限T
	int T = 0;
	int num = 0;
	int K = static_cast<int>(row * col * 0.1);
	for (int i = 765; i >= 0; --i)
	{
		num += RGBSum[i];
		if (num > K)
		{
			T = i;
			break;
		}
	}
	
	// 計算單通道亮區(qū)平均值
	double Bm = 0.0, Gm = 0.0, Rm = 0.0;
	int count = 0;
	for (int i = 0; i < row; ++i)
	{
		uchar *b = Channel[0].ptr<uchar>(i);
		uchar *g = Channel[1].ptr<uchar>(i);
		uchar *r = Channel[2].ptr<uchar>(i);
		for (int j = 0; j < col; ++j)
		{
			int sum = b[j] + g[j] + r[j];
			if (sum > T)
			{
				Bm += b[j];
				Gm += g[j];
				Rm += r[j];
				count++;
			}
 
		}
	}
	Bm /= count;
	Gm /= count;
	Rm /= count;
 
	// 通道調(diào)整
	Channel[0] *= maxB / Bm;
	Channel[1] *= maxG / Gm;
	Channel[2] *= maxR / Rm;
 
	// 合并通道
	cv::merge(Channel, result);
 
	return result;
}
 
int main()
{
	// 載入原圖
	cv::Mat src = cv::imread("test21.jpg");
 
	// 白平衡-完美反射
	cv::Mat result = WhiteBalcane_PRA(src);
 
	// 顯示
	cv::imshow("src", src);
	cv::imshow("result", result);
	cv::waitKey(0);
 
	return 0;
}

測試效果

圖1 原圖

圖2 白平衡后圖像

如圖1所示,是傍晚的一張圖像,眾所周知,傍晚的色溫是較低的,此時采用高于傍晚色溫的色溫值拍照,就會得到一張暖色系的圖片,偏黃;對其進(jìn)行白平衡,使圖片顏色回歸真實的環(huán)境色溫,就得到如圖2的效果。

如果你用過灰度世界算法,你會發(fā)現(xiàn)完美反射算法的效果更亮些;這是因為藍(lán)通道最大值和藍(lán)通道亮區(qū)平均值的差距較大,因而補償?shù)膹姸群軓?;若原圖中不存在白色區(qū),那藍(lán)通道的補償就會較弱,達(dá)不到較好的預(yù)期,因此該算法所處理的圖像中最好有較白的點。

圖3 單色原圖

圖4 單色圖白平衡效果

如圖3所示,是一張色彩相對一致的圖像,整體呈粉色系?;叶仁澜绶▽⑷ǖ罃?shù)值進(jìn)行平均再補償,就會使三通道的數(shù)值趨近一致,進(jìn)而呈現(xiàn)灰色;而完美反射算法,計算的是三通道各自數(shù)值最大值和其亮區(qū)平均值的差距,再進(jìn)行補償,因此三通道的數(shù)值都會適當(dāng)加強,使光感更強,即圖片更亮。

接下來做個有趣的測試,將原本粉色的墻紙設(shè)為較純的綠色。

圖5 調(diào)色后的圖像

圖6 白平衡效果圖

如圖5所示,因為圖像中存在色調(diào)相沖的兩個部分,完美反射算法的優(yōu)勢在這種場景下體現(xiàn)的很明顯。因為綠色區(qū)域較大,灰度世界算法中單純的計算均值再平衡,會無腦地將整圖的綠色分量降低,紅色分量提高,使得結(jié)果異?;?;而完美反射算法,因為圖中有白值,三通道的最大值均在250左右,對綠色分量而言,其最亮區(qū)的均值接近于230,而紅色分量和藍(lán)色分量而言,其最亮區(qū)的均值也接近于200多,因此三通道的平衡結(jié)果就是整體提亮,而不是被無腦平均。

灰度世界算法效果圖見:

C++ OpenCV實現(xiàn)白平衡之灰度世界算法

到此這篇關(guān)于C++ OpenCV實現(xiàn)白平衡之完美反射算法的文章就介紹到這了,更多相關(guān)C++ OpenCV完美反射算法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評論