博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
OpenCV 2.4+ C++ 平滑处理
阅读量:6691 次
发布时间:2019-06-25

本文共 4568 字,大约阅读时间需要 15 分钟。

hot3.png

原理

平滑也称模糊, 是一项简单且使用频率很高的图像处理方法。

平滑处理时需要用到一个滤波器。 最常用的滤波器是线性滤波器,线性滤波处理的输出像素值(例如:g(i,j))是输入像素值(例如:f(i+k,j+l))的加权平均:

    g(i,j) = \sum_{k,l} f(i+k, j+l) h(k,l)

h(k,l)称为核, 它仅仅是一个加权系数。

 

均值平滑

下面是一个使用blur函数的均值平滑:

#include "opencv2/core/core.hpp" #include "opencv2/highgui/highgui.hpp" #include "opencv2/imgproc/imgproc.hpp" #include 
using namespace cv; int main( int argc, char** argv ){ Mat image; image = imread( argv[1]); if( argc != 2 || !image.data ){ printf("没有图片\n"); return -1; } namedWindow( "平滑处理-输入" ); namedWindow( "平滑处理-输出" ); imshow( "平滑处理-输入", image ); Mat out; blur( image, out, Size(3, 3)); imshow( "平滑处理-输出", out ); waitKey( 0 ); }

blur函数API资料:

使用归一化块滤波器进行模糊图片操作。

C++: void blur(InputArray 
src, OutputArray 
dst, Size 
ksize, Point 
anchor=Point(-1,-1), int
borderType=BORDER_DEFAULT )
参数
  • src – 输入图片,可以使是任意通道数,该函数对通道是独立处理的,但是深度只能是CV_8U,CV_16U, CV_16S, CV_32F or CV_64F。
  • dst – 输出图片,和输入图片相同大小和深度。
  • ksize – 模糊内核大小。
  • anchor – 锚点,默认值是(-1,-1),也就是锚点在内核的中心。
  • borderType – 用于判断图像边界的模式。

该函数对图片进行平滑处理利用了下面的内核:

\texttt{K} =  \frac{1}{\texttt{ksize.width*ksize.height}} \begin{bmatrix} 1 & 1 & 1 &  \cdots & 1 & 1  \\ 1 & 1 & 1 &  \cdots & 1 & 1  \\ \hdotsfor{6} \\ 1 & 1 & 1 &  \cdots & 1 & 1  \\ \end{bmatrix}

调用blur(src, dst, ksize, anchor, borderType)相当于调用boxFilter(src,dst, src.type(), anchor, true, borderType)。

blur使用的是归一化块滤波器,输出像素值是核窗口内像素值的均值( 所有像素加权系数相等)。

 

高斯平滑

下面代码使用了GaussianBlur来实现平滑:

#include "opencv2/core/core.hpp" #include "opencv2/highgui/highgui.hpp" #include "opencv2/imgproc/imgproc.hpp" #include 
using namespace std; using namespace cv; int main( int argc, char** argv ){ Mat image; image = imread( argv[1]); if( argc != 2 || !image.data ){ printf("没有图片\n"); return -1; } namedWindow( "平滑处理-输入" ); namedWindow( "平滑处理-输出" ); imshow( "平滑处理-输入", image ); Mat out; GaussianBlur( image, out, Size( 3, 3 ), 0, 0 ); imshow( "平滑处理-输出", out ); waitKey( 0 ); }

GaussianBlur函数API资料:

使用高斯滤波器进行模糊操作

C++: void GaussianBlur(InputArray 
src, OutputArray 
dst, Size 
ksize, double 
sigmaX, double 
sigmaY=0, int 
borderType=BORDER_DEFAULT)
参数
  • src – 输入图片,可以使是任意通道数,该函数对通道是独立处理的,但是深度只能是CV_8U, CV_16U, CV_16S, CV_32F or CV_64F.
  • dst – 输出图片,和输入图片相同大小和深度。
  • ksize – 高斯内核大小。ksize.width和ksize.height允许不相同但他们必须是正奇数。或者等于0,由参数sigma的乘机决定。
  • sigmaX – 高斯内核在X方向的标准偏差。
  • sigmaY – 高斯内核在Y方向的标准偏差。如果sigmaY为0,他将和sigmaX的值相同,如果他们都为0,那么他们由ksize.width和ksize.height计算得出。
  • borderType – 用于判断图像边界的模式。

最有用的滤波器 (尽管不是最快的)。 高斯滤波是将输入数组的每一个像素点与高斯内核卷积将卷积和当作输出像素值。

http://www.cnblogs.com/http://www.cnblogs.com/_images/Smoothing_Tutorial_theory_gaussian_0.jpg

参考一维高斯函数,我们可以看见,他是个中间大两边小的函数。

所以高斯滤波器其加权数是中间大,四周小的。

其二维高斯函数为:

    G_{0}(x, y) = A  e^{ \dfrac{ -(x - \mu_{x})^{2} }{ 2\sigma^{2}_{x} } +  \dfrac{ -(y - \mu_{y})^{2} }{ 2\sigma^{2}_{y} } } 

其中 \mu 为均值 (峰值对应位置),\sigma 代表标准差 (变量x 和 变量y 各有一个均值,也各有一个标准差)。

 

中值平滑

使用medianBlur执行中值平滑:

#include "opencv2/core/core.hpp" #include "opencv2/highgui/highgui.hpp" #include "opencv2/imgproc/imgproc.hpp" #include 
using namespace std; using namespace cv; int main( int argc, char** argv ){ Mat image; image = imread( argv[1]); if( argc != 2 || !image.data ){ printf("没有图片\n"); return -1; } namedWindow( "平滑处理-输入" ); namedWindow( "平滑处理-输出" ); imshow( "平滑处理-输入", image ); Mat out; medianBlur( image, out, 3); imshow( "平滑处理-输出", out ); waitKey( 0 ); }

medianBlur函数API资料:

使用中值滤波器进行模糊操作

C++: void medianBlur(InputArray 
src, OutputArray 
dst, int 
ksize)
Parameters:
  • src – 支持1、3、4通道图片输入,当ksize为3或者5时,图片的深度只能是CV_8U,,CV_16U,或者 CV_32F,对于其他大孔径尺寸只支持深度为CV_8U。
  • dst – 输出图片,和输入图片相同大小和深度。
  • ksize – 线性直径大小,只能是一个大于1的奇数,例如:3, 5, 7 ...

中值滤波将图像的每个像素用邻域 (以当前像素为中心的正方形区域)像素的中值代替 。 

 

双边平滑

使用bilateralFilter执行双边平滑:

#include "opencv2/core/core.hpp" #include "opencv2/highgui/highgui.hpp" #include "opencv2/imgproc/imgproc.hpp" #include 
using namespace std; using namespace cv; int main( int argc, char** argv ){ Mat image; image = imread( argv[1]); if( argc != 2 || !image.data ){ printf("没有图片\n"); return -1; } namedWindow( "平滑处理-输入" ); namedWindow( "平滑处理-输出" ); imshow( "平滑处理-输入", image ); Mat out; bilateralFilter ( image, out, 3, 3*2, 3/2 ); imshow( "平滑处理-输出", out ); waitKey( 0 ); }

bilateralFilter的API资料:

对一个图片应用双边滤波器。

C++: void bilateralFilter(InputArray 
src, OutputArray 
dst, int 
d, double 
sigmaColor, double 
sigmaSpace, int
borderType=BORDER_DEFAULT )
Parameters:
  • src – 源必须是8位或者浮点数,1或者3通道图片。
  • dst – 输出图片,和输入图片相同大小和深度。
  • d – 在滤波过程中使用的各像素邻域直径,如果这是一个非整数,则这个值由sigmaSpace决定。
  • sigmaColor – 颜色空间的标准方差。数值越大,意味着越远的的颜色会被混进邻域内,从而使更大的颜色段获得相同的颜色。
  • sigmaSpace – 坐标空间的标注方差。 数值越大,以为着越远的像素会相互影响,从而使更大的区域足够相似的颜色获取相同的颜色。当d>0,d指定了邻域大小且与sigmaSpace无关。否则,d正比于sigmaSpace。

原理可参考:

目前我们了解的滤波器都是为了平滑图像, 问题是有些时候这些滤波器不仅仅削弱了噪声, 连带着把边缘也给磨掉了。 为避免这样的情形 (至少在一定程度上 ), 我们可以使用双边滤波。 

类似于高斯滤波器,双边滤波器也给每一个邻域像素分配一个加权系数。 这些加权系数包含两个部分, 第一部分加权方式与高斯滤波一样,第二部分的权重则取决于该邻域像素与当前像素的灰度差值。

 

被山寨的原文

 . OpenCV.org

 . OpenCV.org

转载于:https://my.oschina.net/u/218425/blog/91963

你可能感兴趣的文章
关于vs2010巨慢(cpu占用高)的几种解决方式
查看>>
简单3步,轻松集成Testlink和MantisBT
查看>>
Nginx 教程- 获取真实IP模块 - http_realip_module
查看>>
SQL语句教程(04) AND OR
查看>>
EBS 12.1.3 db 11.2.3 dg AND DG SWITCH OVER
查看>>
Oracle中的JOIN
查看>>
html中iframe控制父页面刷新
查看>>
每天一个linux命令(50):crontab命令
查看>>
linux命令7--cat命令&nl命令
查看>>
.NET底层开发技术
查看>>
RHEL regiester
查看>>
c/c++中的一些基础知识
查看>>
练习:输出整数每一位,计算算数,9出现次数,输出图案,水仙花数
查看>>
操作系统的发展
查看>>
HEVC码流简单分析
查看>>
搭建蚂蚁笔记(服务器)
查看>>
lnmp
查看>>
二分查找
查看>>
Cloud Test 在手,宕机时让您不再措手不及
查看>>
Centos7.2安装Vmware Tools
查看>>