一、概要:
使用Canny边缘检测算法作为例子,介绍图像的平滑方法和边缘检测。
Canny边缘检测算法分为四步:
step1:用高斯滤波器平滑图像; step2:用一阶偏导的有限差分来计算梯度的幅值和方向;(在横竖两个方向上计算边缘,再求平方和的开方) step3:对梯度幅值进行非极大值抑制; step4:用双阈值算法检测和连接边缘。
demo&效果:
原图(lenna.bmp):
高斯滤波后的图像:
初步求边缘后的图像:
非极大值抑制后的图像:
双阈值检测后的图像:
以下对这四步进行详细介绍:
二、图像平滑与高斯滤波
图像平滑的目的是消除或尽量减少噪声的影响,改善图像的质量。在假定加性噪声是随机独立分布的条件下,利用邻域的平均或加权平均可以有效的抑制噪声干扰。
方法是做邻域计算,比如:
高斯滤波:
采用高斯函数作为加权函数。 原因一:二维高斯函数具有旋转对称性,保证滤波时各方向平滑程度相同; 原因二:离中心点越远权值越小。确保边缘细节不被模糊。
设定σ2和n,确定高斯模板权值。如σ2 =2和n=5,整数化和归一化后得:
代码:
1.直接用系统函数的方法
img=imread('lenna.bmp'); imshow(img); [m n]=size(img); img=double(img); %高斯滤波 w=fspecial('gaussian',[5 5]); %[ 5 5 ]是模板尺寸,默认是[ 3 3 ] 模板即上文所提的模板 img=imfilter(img,w,'replicate'); figure; imshow(uint8(img))2.自己实现的高斯滤波(使用上面的模板)
img=imread('lenna.bmp'); img=im2double(img); [m n]=size(img); h=zeros(m,n); for i=1:m for j=1:n if i<3 || i>(m-3) || j<3 || j>(n-3) h(i,j)=img(i,j); else %代码长度原因,分行相加。 h(i,j)=h(i,j)+img(i-2,j-2)+img(i-2,j+2)+img(i+2,j-2)+img(i+2,j+2); h(i,j)=h(i,j)+(img(i-1,j-2)+img(i+1,j-2)+img(i-2,j-1)+img(i+2,j-1)+img(i-2,j+1)+img(i+2,j+1)+img(i-1,j+2)+img(i+1,j+2))*2; h(i,j)=h(i,j)+3*(img(i,j-2)+img(i,j+2)+img(i+2,j)+img(i-2,j)); h(i,j)=h(i,j)+4*(img(i-1,j-1)+img(i-1,j+1)+img(i+1,j-1)+img(i+1,j+1)); h(i,j)=h(i,j)+6*(img(i,j-1)+img(i,j+1)+img(i+1,j)+img(i-1,j)); h(i,j)=h(i,j)+7*img(i,j); h(i,j)=h(i,j)/79; end end end imshow(h,[]);
三、初步边缘检测
对高斯平滑后的图像进行sobel边缘检测。这里需要求横的和竖的还有联合的,所以一共三个需要sobel边缘检测图像。
代码:
img=imread('lenna.bmp'); [m n]=size(img); img=double(img); w=fspecial('gaussian',[5 5]); img=imfilter(img,w,'replicate'); figure; %%sobel边缘检测 w=fspecial('sobel'); img_w=imfilter(img,w,'replicate'); %求横边缘 w=w'; %转置矩阵 img_h=imfilter(img,w,'replicate'); %求竖边缘 img=sqrt(img_w.^2+img_h.^2); %平方和再开方 .^表示每一位都和自己乘,不清楚的自己百度 figure; imshow(uint8(img))
四、非极大值抑制以及双阈值检测边缘
什么是非极大值抑制?
非极大值抑制是在梯度方向上的极大值。
代码:(觉得看看注释就差不多明白思路了)
img=imread('lenna.bmp'); imshow(img); [m n]=size(img); img=double(img); %%canny边缘检测的前两步相对不复杂,所以我就直接调用系统函数了 %%高斯滤波 w=fspecial('gaussian',[5 5]); img=imfilter(img,w,'replicate'); figure; imshow(uint8(img)) %%sobel边缘检测 w=fspecial('sobel'); img_w=imfilter(img,w,'replicate'); %求横边缘 w=w'; img_h=imfilter(img,w,'replicate'); %求竖边缘 img=sqrt(img_w.^2+img_h.^2); %注意这里不是简单的求平均,而是平方和在开方。我曾经好长一段时间都搞错了 figure; imshow(uint8(img)) %%下面是非极大抑制 new_edge=zeros(m,n); for i=2:m-1 for j=2:n-1 Mx=img_w(i,j); My=img_h(i,j); if My~=0 o=atan(Mx/My); %边缘的法线弧度 elseif My==0 && Mx>0 o=pi/2; else o=-pi/2; end %Mx处用My和img进行插值 adds=get_coords(o);%边缘像素法线一侧求得的两点坐标,插值需要 M1=My*img(i+adds(2),j+adds(1))+(Mx-My)*img(i+adds(4),j+adds(3)); %插值后得到的像素,用此像素和当前像素比较 adds=get_coords(o+pi);%边缘法线另一侧求得的两点坐标,插值需要 M2=My*img(i+adds(2),j+adds(1))+(Mx-My)*img(i+adds(4),j+adds(3));%另一侧插值得到的像素,同样和当前像素比较 isbigger=(Mx*img(i,j)>M1)*(Mx*img(i,j)>=M2)+(Mx*img(i,j)<=M2); %如果当前点比两边点都大置1 if isbigger new_edge(i,j)=img(i,j); end end end figure; imshow(uint8(new_edge)) %%下面是滞后阈值处理 up=120; %上阈值 low=100; %下阈值 set(0,'RecursionLimit',10000); %设置最大递归深度 for i=1:m for j=1:n if new_edge(i,j)>up &&new_edge(i,j)~=255 %判断上阈值 new_edge(i,j)=255; new_edge=connect(new_edge,i,j,low); end end end figure; imshow(new_edge==255)
get_coords.m:
function re=get_coords(angle) %angle是边缘法线角度,返回法线前后两点 sigma=0.000000001; x1=ceil(cos(angle+pi/8)*sqrt(2)-0.5-sigma); y1=ceil(-sin(angle-pi/8)*sqrt(2)-0.5-sigma); x2=ceil(cos(angle-pi/8)*sqrt(2)-0.5-sigma); y2=ceil(-sin(angle-pi/8)*sqrt(2)-0.5-sigma); re=[x1 y1 x2 y2]; endconnect.m:
function nedge=connect(nedge,y,x,low) %种子定位后的连通分析 neighbour=[-1 -1;-1 0;-1 1;0 -1;0 1;1 -1;1 0;1 1]; %八连通搜寻 [m n]=size(nedge); for k=1:8 yy=y+neighbour(k,1); xx=x+neighbour(k,2); if yy>=1 &&yy<=m &&xx>=1 && xx<=n if nedge(yy,xx)>=low && nedge(yy,xx)~=255 %判断下阈值 nedge(yy,xx)=255; nedge=connect(nedge,yy,xx,low); end end end end
参考资料:
1.http://www.cnblogs.com/tiandsp/archive/2012/12/13/2817240.html
2.南京大学软件学院数字图像处理课程PPT
一些更详细的原理性的东西可以参考:
http://blog.csdn.net/likezhaobin/article/details/6892176