高斯滤波(C++实现)

一维高斯函数:
G ( x ) = 1 2 π σ e − x 2 2 σ 2 G(x)=\frac{1}{\sqrt{2\pi}\sigma} e^{-\frac{x^2}{2\sigma^2}} G(x)=2π σ1e2σ2x2
二维高斯函数:
G ( x , y ) = 1 2 π σ 2 e − x 2 + y 2 2 σ 2 G(x,y)=\frac{1}{2\pi \sigma^2} e^{-\frac{x^2+y^2}{2\sigma^2}} G(x,y)=2πσ21e2σ2x2+y2

下面我们计算高斯核用的函数是:
G ( x , y ) = e − x 2 + y 2 2 σ 2 G(x,y)=e^{-\frac{x^2+y^2}{2\sigma^2}} G(x,y)=e2σ2x2+y2

double** myGetGaussianKernel(int size, double sigma)
{
	double sum = 0.0;
	int center = size/2; // 用于计算中心点的坐标

	double** arr = new double*[size];
	for(int i=0; i<size; ++i){
		arr[i] = new double[size];
	}
	for(int i=0; i<size; ++i){
		for(int j=0; j<size; ++j){
			arr[i][j] = exp(- ((i-center)*(i-center)+(j-center)*(j-center)) / (2*sigma*sigma) )
			sum+=arr[i][j];
		}
	}
	for(int i=0; i<size; ++i){
		for(int j=0; j<size; ++j){
			arr[i][j] /= sum;
		}
	}
	return arr;
}

// 调用OpenCV函数
// 生成一维高斯核
Mat gaus= getGaussianKernel(int size, double sigma, int ktype=CV_64F); 

// 生成二维高斯核(等价于上述函数)
Mat kernel=getGaussianKernel(int size,double sigma, int ktype=CV_64F);
kernel = kernel * kernel.t();

// 将数组赋值给Mat
double guas[3];
Mat kernel=Mat(3,1,CV_64F, gaus).clone();

二维高斯函数可以看成两个一维高斯函数的乘积。
在这里插入图片描述

// C++高斯滤波
void myGaussianBlur(const Mat _src, Mat& _dst, int size, double sigma)
{
	if(!_src.data) return;
	Mat src = _src.clone();
	Mat dst(src.size(), src.type(), Scalar::all(0));
	double** gaus = myGetGaussianKernel(size, sigma);
	// 等价于
	// Mat_<double> gaus = getGaussianKernel(3,1,CV_64F);
	// gaus = _gaus * _gaus.t();
	
	int center = size/2;

	copyMakeBorder(_src, src, center, center, center, center, BORDER_REPLICATE);

	for(int i=center; i<src.rows-center; ++i){
		for(int j=center; j<src.cols-center; ++j){
			for(int x=-center; x<=center; ++x){
				for(int y=-center; y<=center; ++y){
					dst.at<Vec3b>(i-center, j-center)[0] += src.at<Vec3b>(i+x, j+y)[0] * gaus[x+center][y+center];
					dst.at<Vec3b>(i-center, j-center)[1] += src.at<Vec3b>(i+x, j+y)[1] * gaus[x+center][center];
					dst.at<Vec3b>(i-center,j-center)[2] += src.at<Vec3b>(i+x, j+y)[2] * gaus[x+center][center];
				}
			}
		}
	}
	dst.copyTo(_dst);
}

参考资料:
[1] openCV之高斯滤波(及代码实现)
[2] OpenCV 源码分析-getGaussianKernel
[3] 基于C++&OpenCV实现高斯滤波【数字图像处理】
[4] 高斯滤波的C++实现与优化

  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值