空间域图像增强:OpenCV图像积分图算法
0.积分图算法介绍
积分图(integral image
)算法是图像处理中的经典算法之一,由Crow在1984年首次提出,它是为了在多尺度透视投影中提高渲染速度。
积分图算法是一种快速计算图像区域和以及图像区域平方和的算法。直白的说,就是很快计算一幅图像任意区域,也就是卷积区域下的像素值的和跟平方和。它的核心思想就是对每一个图像建立起自己的积分图查找表,在图像处理的阶段就可以根据预先建立积分图查找表直接查找从而实现对均值卷积的线性时间计算,做到了卷积执行的时间与半径窗口大小的无关联。
这种算法被应用到基于NCC
的快速匹配、对象检测和SURF
变换中、基于统计学的快速滤波器等方面。第一个应用积分图像技术的应用是在Viola-Jones
的对象检测框架中出现。
1.图像积分图的建立与查找
1.1积分图的建立
图像积分图是根据原图像像素值而计算建立出来的,假设原图的大小为 W ∗ H W*H W∗H,则积分图的大小为 ( W + 1 ) ∗ ( H + 1 ) (W+1)*(H+1) (W+1)∗(H+1)。在积分图上任意坐标 ( x , y ) (x,y) (x,y)处的 i i ( x , y ii(x,y ii(x,y)即表示原图中坐标为 ( x , y ) (x,y) (x,y)的点的左上角所有像素点像素值的和(平方和表中的就是平方和)。
和表和平方和表建立公式如下:
和表:
平方和表:
举个栗子,假设输入图像为2x2大小的,则积分图为3x3大小。如下图:
1.2积分图的查找
如上图,根据输入图像得到该图像的积分图之后,如果想求输入图像中的蓝色区域内的像素值之和:
3
+
2
+
5
+
4
=
14
3+2+5+4=14
3+2+5+4=14
只要根据每个点左上方所有像素值和表值进行两次减法和一次减法即可:
46
−
22
−
20
+
10
=
14
46-22-20+10=14
46−22−20+10=14
也就是右下角+左上角-右上角和左下角。
这样对于任意大小的区域,实际上只要进行两次减法和一次加法就可以计算出区域内像素值之和。
2.OpenCV积分图api
OpenCV
中计算积分图的C++ api
原型如下:
void integral( InputArray src, //输入图像
OutputArray sum,//和表
OutputArray sqsum,//平方和表
int sdepth = -1, //和表深度
int sqdepth = -1 ); //平方和表深度
其中 src
为输入图像, sum
就是和表, sqsum
是平方和表, sdepth
是和表深度,一般用 CV_32S
, sqdepth
是平方和表深度,一般为 CV_32F
。
举个栗子,获取输入图像的积分图代码如下:
Mat input_image = imread("./whisper.jpg", 1);
if (input_image.empty())
{
cout << "read input error!" << endl;
return -1;
}
imshow("input", input_image);
Mat sum, sqrsum;
integral(input_image, sum, sqrsum, CV_32S, CV_32F);
则sum
和sqrsum
即分别为和表和平方和表。得到积分和图表后就可以根据上面的积分图查表方法编写get_block_sum()
函数来获取指定区域内像素点的像素值之和:
int get_block_sum(Mat &sum, int x1, int y1, int x2, int y2, int i)
{
int tl = sum.at<Vec3i>(y1, x1)[i];
int tr = sum.at<Vec3i>(y2, x1)[i];
int bl = sum.at<Vec3i>(y1, x2)[i];
int br = sum.at<Vec3i>(y2, x2)[i];
int sum_value = (br - bl - tr + tl);
return sum_value;
}