百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分析 > 正文

调节彩色图像亮度(基于不同的颜色空间)

liebian365 2024-10-27 13:18 21 浏览 0 评论

使用C++、opencv基于不同的颜色空间调节图像亮度

该博客简单介绍了RGB、YUV、HSI等颜色空间的相关知识,并附RGB转到另两个颜色空间的转换公式: https://www.cnblogs.com/justkong/p/6570914.html

https://blog.csdn.net/sinat_26917383/article/details/70860910及https://wenku.baidu.com/view/6273f20a581b6bd97f19ea99.html也都介绍了不同的颜色空间的相关知识。

该博客介绍了opencv中颜色转换的函数及RGBHSV空间的实际应用:https://blog.csdn.net/dieju8330/article/details/82465616


:opencv中HSI、HSV、HSL中的H分量的范围是0-180,剩下两个(SI、SL、SV)都是0-255;

V分量即是RGB中最大的那个分量,与图像亮度无关,所以选择HSL空间研究;

opencv中颜色空间转换函数在2.x与3.x版本中都是cvtColor(InputArray src, OutputArray dst, int code, int dstCn=0),只是code的宏定义的写法有点区别(3.x中将2.x中的CV前缀变为COLOR前缀)。


该博客介绍了HSLHSV颜色空间的区别:https://blog.csdn.net/binglan520/article/details/56288135

注:在公式上HSL、HSV、HSI的区别在于:L=(MAX(R,G,B)+MIN(R,G,B))/2;V=MAX(R,G,B));I=(R+G+B)/3。


该博客介绍了不同光照条件(即亮度条件)下几个颜色空间的色彩分量的区别及应用颜色阈值进行分割的相关知识:https://blog.csdn.net/wc781708249/article/details/78517463


本代码在不同颜色空间下(HSI、Lab、YCrCb),对跟亮度有关的颜色分量(I、L、Y)进行直方图均衡化,达到平衡亮度的效果,其次以YCrCb空间为例,展示利用滑动条调节图像整体Y值。

相关重要API:

void cvtColor(InputArray src, OutputArray dst, int code, int dstCn=0 );

颜色空间转换函数

InputArray src: 输入图像即要进行颜色空间变换的原图像,可以是Mat类;

OutputArray dst: 输出图像即进行颜色空间变换后存储图像,也可以Mat类;

int code: 转换的代码或标识,即在此确定将什么制式的图片转换成什么制式的图片,具体见下图;

int dstCn = 0: 目标图像通道数,如果取值为0,则由src和code决定


void split(const Mat& src, Mat*mvbegin);

void split(InputArray m,OutputArrayOfArrays mv);

split()函数可以把一幅图像各个通道分离开

第一个参数为要进行分离的图像矩阵,第二个参数可以是Mat数组的首地址,或者一个vector<Mat>对象

void merge(const Mat* mv, size_tcount, OutputArray dst)
void merge(InputArrayOfArrays mv,OutputArray dst)

经过对各个通道单独操作后可以用merge()函数进行合并

第一种:第一个参数是图像矩阵数组,第二个参数是需要合并矩阵的个数,第三个参数是输出。

第二种:第一个参数是图像矩阵向量容器,第二个参数是输出,这种方法无需说明需要合并的矩阵个数,vector对象自带说明。

以上部分参考:https://blog.csdn.net/keith_bb/article/details/53470170、https://blog.csdn.net/u012819339/article/details/82222008、https://blog.csdn.net/xiexu911/article/details/79722535


#include "stdafx.h"
#include "opencv2/imgproc/imgproc.hpp"
#include "opencv2/highgui/highgui.hpp"
#include <iostream>
#include <fstream>
#include <iterator>

using namespace cv;
using namespace std;

//定义一些变量及声明函数
Mat src = imread("C:\\Users\\lenovo\\Desktop\\1.jpg");

Mat hsv, lab, ycrcb;
Mat hsvdst, labdst, ycrcbdst;
Mat dst(src.size(), src.type(), Scalar::all(0));
void balance_hsv(Mat & hsv);
void balance_lab(Mat & lab);
void balance_ycrcb(Mat & ycrcb);
void drawHist(Mat cur);

int y_value = 0;
void on_changebright(int, void *);

int main(int argc, char** argv)
{
	system("color 02");
	/*//获取图像每个像素点的RGBHSVLabYCrCb的值,并写入文件中
	cvtColor(src, hsv, COLOR_BGR2HSV);
	cvtColor(src, lab, COLOR_BGR2Lab);
	cvtColor(src, ycrcb, COLOR_BGR2YCrCb);
	FILE *fp;
	fopen_s(&fp, "C:\\Users\\lenovo\\Desktop\\RGB数据.xls", "a");
	for(int i=0;i<src.rows;i++)
		for (int j = 0; j < src.cols; j++)
		{
			fprintf(fp, "%d\t%d\t%d\t%d\t%d\t%d\t%d\t%d\t%d\t%d\t%d\t%d\n", src.at<Vec3b>(i, j)[2], src.at<Vec3b>(i, j)[1], src.at<Vec3b>(i, j)[0],
				hsv.at<Vec3b>(i, j)[0], hsv.at<Vec3b>(i, j)[1], hsv.at<Vec3b>(i, j)[2],
				lab.at<Vec3b>(i, j)[0], lab.at<Vec3b>(i, j)[1], lab.at<Vec3b>(i, j)[2], 
				ycrcb.at<Vec3b>(i, j)[0], ycrcb.at<Vec3b>(i, j)[1], ycrcb.at<Vec3b>(i, j)[2]);
		}
	fclose(fp);*/
	
	//平衡图像亮度
	balance_hsv(src);
	balance_lab(src);
	balance_ycrcb(src);
    //显示源图像
	namedWindow("src", WINDOW_NORMAL);
	imshow("src", src);

    //创建调节Y值的滑动条
    createTrackbar("y", "bright", &y_value, 200,on_changebright);
    //滑动条回调函数
	on_changebright(0, 0);
	waitKey(0);
	return 0;
}

void balance_hsv(Mat & src)
{
    //将源图像转换到HSV颜色空间	
    cvtColor(src, hsv, COLOR_BGR2HSV);
    //分割为3通道
	Mat splitchannels[3], temp;
	split(hsv, splitchannels);

    
	if (splitchannels[2].channels() == 1)
	{
		equalizeHist(splitchannels[2], temp);//直方图均衡化
	}
	/*drawHist(dst);//调用函数画直方图
	imshow("hist", yuvhist);//显示均衡化的直方图*/

	std::vector<Mat> dst_merge;
	dst_merge.push_back(splitchannels[0]);
	dst_merge.push_back(splitchannels[1]);
	dst_merge.push_back(temp);

	//合并修改过的3个通道图像
	merge(dst_merge, dst);
	//转换回RGB空间
	cvtColor(dst, hsvdst, COLOR_HSV2BGR);
	//显示经直方图均衡化的图像
	namedWindow("balance_V", WINDOW_NORMAL);
	imshow("balance_V", hsvdst);

}

void balance_lab(Mat & src)
{
    //将源图像转换到Lab颜色空间
	cvtColor(src, lab, COLOR_BGR2Lab);
	Mat splitchannels[3], temp;

	split(lab, splitchannels);

	if (splitchannels[0].channels() == 1)
	{
		equalizeHist(splitchannels[0], temp);//直方图均衡化
	}
	/*drawHist(dst);//调用函数画直方图
	imshow("hist", yuvhist);//显示均衡化的直方图*/

	std::vector<Mat> dst_merge;
	dst_merge.push_back(temp);
	dst_merge.push_back(splitchannels[1]);
	dst_merge.push_back(splitchannels[2]);
	
	//合并修改过的3个通道图像
	merge(dst_merge, dst);
	//转换回RGB空间
	cvtColor(dst, labdst, COLOR_Lab2BGR);
	//显示经直方图均衡化的图像
	namedWindow("balance_L", WINDOW_NORMAL);
	imshow("balance_L", labdst);

}

void balance_ycrcb(Mat & src)
{
	//将源图像转换到YCrCb颜色空间
    cvtColor(src, ycrcb, COLOR_BGR2YCrCb);
	Mat splitchannels[3], temp;

	split(ycrcb, splitchannels);

	if (splitchannels[0].channels() == 1)
	{
		equalizeHist(splitchannels[0], temp);//直方图均衡化
	}
	/*drawHist(dst);//调用函数画直方图
	imshow("hist", yuvhist);//显示均衡化的直方图*/

	std::vector<Mat> dst_merge;
	dst_merge.push_back(temp);
	dst_merge.push_back(splitchannels[1]);
	dst_merge.push_back(splitchannels[2]);
	
	//合并修改过的3个通道图像
	merge(dst_merge, dst);
	//转换回RGB空间
	cvtColor(dst, ycrcbdst, COLOR_YCrCb2BGR);
	//显示经直方图均衡化的图像
	namedWindow("balance_Y", WINDOW_NORMAL);
	imshow("balance_Y", ycrcbdst);

}

void on_changebright(int, void *)
{
	Mat src1;
    //将源图拷贝至目标图实现变化实时更新
	dst.copyTo(src1);
    //转换至YCrCb颜色空间
	cvtColor(src1, src1, COLOR_BGR2YCrCb);
    //调节每个点的Y值,这里也可以换成其他变化,如降低图像的亮度值或其他颜色值
	for(int i=0;i<src1.rows;i++)
		for (int j = 0; j < src1.cols; j++)
		{
			//Y值增加一个给定值
            src1.at<Vec3b>(i, j)[2] += y_value;
            //溢出保护
			if (src1.at<Vec3b>(i, j)[2] > 255 )
				src1.at<Vec3b>(i, j)[2] = 255;
		}
    //转换回RGB空间观察变化
	cvtColor(src1, src1, COLOR_HSV2BGR);
    
    //显示亮度调节后的图像
    namedWindow("bright", WINDOW_NORMAL);
	imshow("bright", src1);

}

void drawHist(Mat cur)
{
	//需要计算图像的哪个通道(bgr空间需要确定计算 b或g或r空间)      
	const int channels[1] = { 0 };

	//直方图的每一个维度的 柱条的数目(就是将灰度级分组)    
	int histSize[] = { 256 };   //如果这里写成int histSize = 256;   那么下面调用计算直方图的函数的时候,该变量需要写 &histSize    

	//定义一个变量用来存储 单个维度 的数值的取值范围      
	float midRanges[] = { 0, 256 };

	//确定每个维度的取值范围,就是横坐标的总数      
	const float *ranges[] = { midRanges };

	//输出的结果存储的 空间 ,用MatND类型来存储结果    
	MatND dstHist;

	calcHist(&cur, 1, channels, Mat(), dstHist, 1, histSize, ranges, true, false);

	//calcHist  函数调用结束后,dstHist变量中将储存了 直方图的信息  用dstHist的模版函数 at<Type>(i)得到第i个柱条的值  at<Type>(i, j)得到第i个并且第j个柱条的值      

	//首先先创建一个白底的图像,为了可以显示彩色,所以该绘制图像是一个8位的3通道图像      
	yuvhist = Mat(Size(128, 128), CV_8UC3, Scalar::all(255));

	//一个图像的某个灰度级的像素个数(最多为图像像素总数),可能会超过显示直方图的所定义的图像的尺寸,因此绘制直方图的时候,让直方图最高的地方只有图像高度的90%来显示    

	//先用minMaxLoc函数来得到计算直方图后的像素的最大个数      
	double g_dHistMaxValue;
	minMaxLoc(dstHist, 0, &g_dHistMaxValue, 0, 0);

	//遍历直方图得到的数据      
	for (int i = 0; i < 128; i++)
	{
		int value = cvRound(128 * 0.9 *(dstHist.at<float>(i) / g_dHistMaxValue));
		line(yuvhist, Point(i, yuvhist.rows - 1), Point(i, yuvhist.rows - 1 - value), Scalar(0, 255, 50));
	}
}

源图像:

?

平衡V后的图像:

?

平衡L后的图像:

?

平衡Y后的图像:

?

调节Y值的图像(其他变化,如降低图像的亮度值或其他颜色值等可自行根据需求尝试):

?

画出的直方图就不演示了。

相关推荐

4万多吨豪华游轮遇险 竟是因为这个原因……

(观察者网讯)4.7万吨豪华游轮搁浅,竟是因为油量太低?据观察者网此前报道,挪威游轮“维京天空”号上周六(23日)在挪威近海发生引擎故障搁浅。船上载有1300多人,其中28人受伤住院。经过数天的调...

“菜鸟黑客”必用兵器之“渗透测试篇二”

"菜鸟黑客"必用兵器之"渗透测试篇二"上篇文章主要针对伙伴们对"渗透测试"应该如何学习?"渗透测试"的基本流程?本篇文章继续上次的分享,接着介绍一下黑客们常用的渗透测试工具有哪些?以及用实验环境让大家...

科幻春晚丨《震动羽翼说“Hello”》两万年星间飞行,探测器对地球的最终告白

作者|藤井太洋译者|祝力新【编者按】2021年科幻春晚的最后一篇小说,来自大家喜爱的日本科幻作家藤井太洋。小说将视角放在一颗太空探测器上,延续了他一贯的浪漫风格。...

麦子陪你做作业(二):KEGG通路数据库的正确打开姿势

作者:麦子KEGG是通路数据库中最庞大的,涵盖基因组网络信息,主要注释基因的功能和调控关系。当我们选到了合适的候选分子,单变量研究也已做完,接着研究机制的时便可使用到它。你需要了解你的分子目前已有哪些...

知存科技王绍迪:突破存储墙瓶颈,详解存算一体架构优势

智东西(公众号:zhidxcom)编辑|韦世玮智东西6月5日消息,近日,在落幕不久的GTIC2021嵌入式AI创新峰会上,知存科技CEO王绍迪博士以《存算一体AI芯片:AIoT设备的算力新选择》...

每日新闻播报(September 14)_每日新闻播报英文

AnOscarstatuestandscoveredwithplasticduringpreparationsleadinguptothe87thAcademyAward...

香港新巴城巴开放实时到站数据 供科技界研发使用

中新网3月22日电据香港《明报》报道,香港特区政府致力推动智慧城市,鼓励公私营机构开放数据,以便科技界研发使用。香港运输署21日与新巴及城巴(两巴)公司签署谅解备忘录,两巴将于2019年第3季度,开...

5款不容错过的APP: Red Bull Alert,Flipagram,WifiMapper

本周有不少非常出色的app推出,鸵鸟电台做了一个小合集。亮相本周榜单的有WifiMapper's安卓版的app,其中包含了RedBull的一款新型闹钟,还有一款可爱的怪物主题益智游戏。一起来看看我...

Qt动画效果展示_qt显示图片

今天在这篇博文中,主要实践Qt动画,做一个实例来讲解Qt动画使用,其界面如下图所示(由于没有录制为gif动画图片,所以请各位下载查看效果):该程序使用应用程序单窗口,主窗口继承于QMainWindow...

如何从0到1设计实现一门自己的脚本语言

作者:dong...

三年级语文上册 仿写句子 需要的直接下载打印吧

描写秋天的好句好段1.秋天来了,山野变成了美丽的图画。苹果露出红红的脸庞,梨树挂起金黄的灯笼,高粱举起了燃烧的火把。大雁在天空一会儿写“人”字,一会儿写“一”字。2.花园里,菊花争奇斗艳,红的似火,粉...

C++|那些一看就很简洁、优雅、经典的小代码段

目录0等概率随机洗牌:1大小写转换2字符串复制...

二年级上册语文必考句子仿写,家长打印,孩子照着练

二年级上册语文必考句子仿写,家长打印,孩子照着练。具体如下:...

一年级语文上 句子专项练习(可打印)

...

亲自上阵!C++ 大佬深度“剧透”:C++26 将如何在代码生成上对抗 Rust?

...

取消回复欢迎 发表评论: