什么是Gamma校正?
Gamma校正是对输入图像灰度值进行的非线性操作,使输出图像灰度值与输入图像灰度值呈指数关系。
Gamma校正的原理表达式如下:
上面中的指数γ即为Gamma。这就是Gamma校正的名称来历。
其中的取值范围是0~1,最重要的参数就是式子中的参数γ。
γ的值决定了输入图像和输出图像之间的灰度映射方式,即决定了是增强低灰度值区域还是增高灰度值区域。
γ>1时,即下图中的红色曲线,图像的高灰度区域对比度得到增强,直观效果是一幅偏亮的图变暗了下来。
γ<1时,即下图中的蓝色曲线,图像的低灰度区域对比度得到增强,直观效果是一幅偏暗的图变亮了起来。
γ=1时,不改变原图像。
Gamma校正表达式的曲线图如下:
横坐标是输入灰度值,纵坐标是输出灰度值,蓝色曲线是gamma值小于1时的输入输出关系,红色曲线是gamma值大于1时的输入输出关系。
可以观察到,当gamma值小于1时(蓝色曲线),图像的整体亮度值得到提升,同时低灰度处的对比度增加,高灰度处的对比度降低,更利于分辩低灰度值时的图像细节;
当gamma值大于1时(红色曲线),图像的整体亮度值得到减小,同时低灰度处的对比度降低,高灰度处的对比度增加,更利于分辩高灰度值时的图像细节。
为什么要进行Gamma校正?
一个典型的例子:人眼对外界光源的感光值与输入光强不是呈线性关系的,而是呈指数型关系的。在低照度下,人眼更容易分辨出亮度的变化,随着照度的增加,人眼不易分辨出亮度的变化。而相机感光与输入光强呈线性关系。为了适应我们人眼的特性,我们就需要对相机拍摄得到的原始图像作Gamma校正。
在OpenCV环境下实现的Gamma校正代码如下:
//出处:昊虹AI笔记网(hhai.cc)
//用心记录计算机视觉和AI技术
//博主微信/QQ 2487872782
//QQ群 271891601
//欢迎技术交流与咨询
//OpenCV版本 OpenCV3.0
#include <opencv2/opencv.hpp>
#include <iostream>
using namespace std;
using namespace cv;
void MyGammaCorrection(Mat& src, Mat& dst, float fGamma)
{
// build look up table
unsigned char lut[256];
for( int i = 0; i < 256; i++ )
{
lut[i] = saturate_cast<uchar>(pow((float)(i/255.0), fGamma) * 255.0f);
}
dst = src.clone();
const int channels = dst.channels();
switch(channels)
{
case 1: //灰度图的情况
{
MatIterator_<uchar> it, end;
for( it = dst.begin<uchar>(), end = dst.end<uchar>(); it != end; it++ )
//*it = pow((float)(((*it))/255.0), fGamma) * 255.0;
*it = lut[(*it)];
break;
}
case 3: //彩色图的情况
{
MatIterator_<Vec3b> it, end;
for( it = dst.begin<Vec3b>(), end = dst.end<Vec3b>(); it != end; it++ )
{
//(*it)[0] = pow((float)(((*it)[0])/255.0), fGamma) * 255.0;
//(*it)[1] = pow((float)(((*it)[1])/255.0), fGamma) * 255.0;
//(*it)[2] = pow((float)(((*it)[2])/255.0), fGamma) * 255.0;
(*it)[0] = lut[((*it)[0])];
(*it)[1] = lut[((*it)[1])];
(*it)[2] = lut[((*it)[2])];
}
break;
}
}
}
int main()
{
Mat image = imread("E:/material/images/P0034-Gamma-correction.jpg");
if (image.empty())
{
cout << "Error: Could not load image" << endl;
return 0;
}
Mat dst1;
Mat dst2;
float fGamma1=1/2.2;
float fGamma2=2;
MyGammaCorrection(image, dst1, fGamma1);
MyGammaCorrection(image, dst2, fGamma2);
cv::namedWindow("Source Image", WINDOW_NORMAL);
cv::namedWindow("Gamma=1/2.2", WINDOW_NORMAL);
cv::namedWindow("Gamma=2", WINDOW_NORMAL);
imshow("Source Image", image);
imshow("Gamma=1/2.2", dst1);
imshow("Gamma=2", dst2);
waitKey();
return 0;
}
从上面的结果中我们可以看出,当γ值小于1时,图像的低灰度区域对比度得到了增强,这样就使得一幅偏暗的图变得明亮了起来;而当γ值大于1时,图像的高灰度区域对比度得到了增强,这就会使得一幅偏亮的图暗下来。
接下来再测试下彩色图片的效果。
用下面这张图进行测试:
上面这张图的百度网盘下载链接:
链接:https://pan.baidu.com/s/1FPaoTbFRKITnVtvFIoL1Yw?pwd=eq8i
运行效果如下图所示: