OpenCV 学习(利用滤波器进行边缘提取)

通过低通滤波器,我们可以将图像平滑,相反的,利用高通滤波器可以提取出图像的边缘。

Sobel 滤波器

Sobel 滤波器是一种有方向性的滤波器,可以作用在 X 方向或 Y 方向。
关于这种滤波器的理论介绍可以参考:

https://en.wikipedia.org/wiki/Sobel_operator

函数原型如下:

void Sobel( InputArray src, OutputArray dst, int ddepth,
                     int dx, int dy, int ksize=3,
                     double scale=1, double delta=0,
                     int borderType=BORDER_DEFAULT );

这个滤波器结合了高斯平滑滤波和差分运算,对噪声不是很敏感,是一种很常用的边缘检测算子。 dx 和 dy 是 X 和 Y 方向差分运算的阶数。
如果对 X 方向求1 阶差分,这对参数设为 1, 0。 对 Y 方向则设为 0, 1。

ksize 是核的大小,只能为 1, 3, 5, 7。 ksize = 1 时核为 1行3列或 3行1列,这时高斯平滑的步骤就没有了。

ksize 还可以设为 CV_SCHARR (-1),这算是个隐藏功能。 这时实际上计算的是 3 * 3 Scharr 滤波器。

scale 是对计算结果的缩放,delta 是对计算结果的平移。

下面是一个例子,原始测试图像如下:

python opencv mask边缘 opencv 边缘提取_边缘提取

这里我们对 X 方向进行边缘检测。因为计算结果会出现负数,所以还做了些放缩和平移操作。代码如下:

cv::Sobel(image, result, CV_8U, 1, 0, 3, 0.5, 128);

处理后的图像如下:

python opencv mask边缘 opencv 边缘提取_边缘检测_02


可以看到图像X方向的正负边缘都检测出来了,并且正负边缘的数值是不同的,这对于我们需要提取某一种特定边缘时是很有利的。

如果我们不需要区分正负边缘,可以取个绝对值运算。类似下面这样。

cv::Sobel(image, result, CV_16S, 1, 0, 3);
result = abs(result);
result.convertTo(result, CV_8U);

这里需要注意的是,当 image 是 CV_8U 类型时,result 类型不能是 CV_8S。这里只能先转成 CV_16S 然后再转换为我们需要的 CV_8U。

python opencv mask边缘 opencv 边缘提取_边缘检测_03

分别对图像的 X 方向和 Y 方向进行 Sobel 滤波后我们就得到了图像的梯度信息。但是这个梯度信息是(X, Y)形式的。有时我们需要用到 (ρ,θ)

cv::Sobel(image, resultX, CV_32F, 1, 0, 3);
cv::Sobel(image, resultY, CV_32F, 0, 1, 3);
cv::Mat norm, dir;
cv::cartToPolar(resultX, resultY, norm, dir);

python opencv mask边缘 opencv 边缘提取_差分_04

Scharr 滤波器

Scharr 算子在图像的梯度方向的计算方面比 Sobel 算子更准确一些。用法和 Sobel 算子是类似的。

具体的理论也可以参考:

https://en.wikipedia.org/wiki/Sobel_operator

下面是 Scharr 算子的函数原型。

void Scharr( InputArray src, OutputArray dst, int ddepth,
                      int dx, int dy, double scale=1, double delta=0,
                      int borderType=BORDER_DEFAULT );

简单的说:

Scharr(src, dst, ddepth, dx, dy, scale, delta, borderType);

等效于:

Sobel(src, dst, ddepth, dx, dy, CV_SCHARR, scale, delta, borderType);