我预期高斯模糊操作是对称的,但是使用OpenCV 2.4.11 GaussianBlur,我得到了不同的结果。
下面是一个例子。我将GaussianBlur应用于图像,并将其应用于图像的翻转版本。我已经单独验证了flip操作不会改变图像像素值(未显示)。当我将模糊图像翻转回时,我希望它与原始图像的模糊程度相同,但是diff显示了许多小的差异(在0.0和6.103515625e-005之间)。我知道这很小,但在我接下来的处理过程中,它会产生连锁反应。
高斯核是对称的,所以结果应该是相同的。这仅仅是实现中的舍入错误吗?
int main(int, char **)
{
// e.g. 2008_005541.jpg from VOC2012 dataset
char const * const filename = "...";
float const sig_diff = 1.24899971f;
cv::Mat image = cv::imread(filename, cv::IMREAD_GRAYSCALE);
cv::Mat gray_fpt;
image.convertTo(gray_fpt, cv::DataType<float>::type, 1, 0);
GaussianBlur(gray_fpt, gray_fpt, cv::Size(), sig_diff, sig_diff);
cv::Mat mirror;
flip(image, mirror, 1);
cv::Mat mirror_gray_fpt;
mirror.convertTo(mirror_gray_fpt, cv::DataType<float>::type, 1, 0);
GaussianBlur(mirror_gray_fpt, mirror_gray_fpt, cv::Size(), sig_diff, sig_diff);
flip(mirror_gray_fpt, mirror_gray_fpt, 1);
cv::Mat diff = abs(gray_fpt - mirror_gray_fpt);
double minval, maxval;
minMaxLoc(diff, &minval, &maxval);
// minval = 0.0;
// maxval = 6.103515625e-005;
// easier to visualise the differences with this:
normalize(diff, diff, 0.0, 1.0, cv::NORM_MINMAX, CV_32FC1);
return 0;
}编辑:我将类型从cv::DataType<float>::type改为cv::DataType<double>::type,现在最大错误为1.1368683772161603e-013,所以舍入似乎是个问题。
发布于 2015-04-24 08:50:00
将上面的代码更改为调用gaussian_blur (下面)而不是GaussianBlur,在我已经测试过的示例映像中没有任何不同。
由此,如果高斯运算的工作类型为double精度,则float定点精度的输出不会产生误差。这似乎是解决我问题的好办法。
// Perform gaussian blur in double precision and convert back
void gaussian_blur(
cv::Mat const &src, cv::Mat &dst,
cv::Size ksize, double sigmaX, double sigmaY=0,
int borderType=cv::BORDER_DEFAULT)
{
cv::Mat src_dp;
src.convertTo(src_dp, cv::DataType<double>::type, SIFT_FIXPT_SCALE, 0);
cv::Mat dst_dp;
GaussianBlur(src_dp, dst_dp, ksize, sigmaX, sigmaY, borderType);
dst_dp.convertTo(dst, src.type(), SIFT_FIXPT_SCALE, 0);
}https://stackoverflow.com/questions/29840429
复制相似问题