深圳幻海软件技术有限公司 欢迎您!

【计算机视觉】图像分割与特征提取——基于Log、Canny的边缘检测

2023-02-28

个人简介: >📦个人主页:赵四司机>🏆学习方向:JAVA后端开发 >⏰往期文章:SpringBoot项目整合微信支付>🔔博主推荐网站:牛客网刷题|面试|找工作神器>📣种一棵树最好的时间是十年前,其次是现在!>💖喜欢的话麻烦点点关注喔,

个人简介: 

> 📦个人主页:赵四司机
> 🏆学习方向:JAVA后端开发 
> ⏰往期文章:SpringBoot项目整合微信支付
> 🔔博主推荐网站:牛客网 刷题|面试|找工作神器
> 📣种一棵树最好的时间是十年前,其次是现在!
> 💖喜欢的话麻烦点点关注喔,你们的支持是我的最大动力。

前言:

实验中要求能够自行评价Log、Canny算子在无噪声条件下和噪声条件下的分割性能。能够掌握分割条件(阈值等)的选择。完成规定图像的处理并要求正确评价处理结果,能够从理论上作出合理的解释。通过实验体会一些主要的分割算子对图像处理的效果,以及各种因素对分割效果的影响。

目录

一:相关概念

1.一阶导数二阶导数在图像边缘中的应用 

2.边缘检测基本步骤

3.好的边缘检测遵循的原则

4.二阶微分(差分)算子

二:拉普拉斯算子

1.等效式

2.拉普拉斯算子

​​​​3.拉普拉斯算子缺点 

三:高斯-拉普拉斯算子(Log)

1.相关概念

2.Log边缘检测

(1)基本特征

(2) 代码实战

(3)结果分析

四:Canny边缘检测

1.相概念

2.检测步骤

3.代码实战

4.结果分析 

五:思考总结


一:相关概念

1.一阶导数二阶导数在图像边缘中的应用 

  • 一阶导数可以用于检测图像中的一个点是否是边缘点(判断一个点是否在斜坡上);
  • 二阶导数可以用于判断一个边缘像素是在边缘亮的一边还是暗的一边。 

2.边缘检测基本步骤

  1. 滤波:边缘检测算法主要是基于图像强度的一阶和二阶导数,但导数的计算对噪声很敏感,因此必须使用滤波器来改善与噪声有关的边缘检测器的性能.需要指出,大多数滤波器在降低噪声的同时也导致了边缘强度的损失,因此,增强边缘和降低噪声之间需要折衷。
  2. 增强:增强边缘的基础是确定图像各点邻域强度的变化值.增强算法可以将邻域(或局部)强度值有显著变化的点突显出来.边缘增强一般是通过计算梯度幅值来完成的。
  3. 检测:在图像中有许多点的梯度幅值比较大,而这些点在特定的应用领域中并不都是边缘,所以应该用某种方法来确定哪些点是边缘点.最简单的边缘检测判据是梯度幅值阈值判据。
  4. 定位:如果某一应用场合要求确定边缘位置,则边缘的位置可在子像素分辨率上来估计,边缘的方位也可以被估计出来 

3.好的边缘检测遵循的原则

  • 有好的检测率:

    •  最优的检测器应该能检测出所有真实边缘,而忽略噪声和其他瑕疵

  • 有好的局部化效果:

    •  检测到的边缘必须尽可能的靠近真实边缘

    • 对于每个真实边缘点,检测器必须返回一个点

4.二阶微分(差分)算子

        如果所求的一阶导数高于某一阈值,则确定该点为边缘点,这样做会导致检测的边缘点太多。 一种更好的方法就是求梯度局部最大值对应的点,并认定它们是边缘点,通过去除一阶导数中的非局部最大值,可以检测出更精确的边缘。 一阶导数的局部最大值对应着二阶导数的零交叉点。这样,通过找图像强度的二阶导数的零交叉点就能找到边缘点。

二:拉普拉斯算子

1.等效式

   

2.拉普拉斯算子

    

  • 二阶微分关心的是图像灰度的突变而不强调灰度缓慢变化的区域,对边缘的定位能力更强。
  •  Laplace算子是各项同性的,即具有旋转不变性

​​​​3.拉普拉斯算子缺点 

  • 拉普拉斯的二阶方向导数算子在机器视觉中并不常用, 因为任何包含有二阶导数的算子比只包含有一阶导数的算子更易受噪声的影响.甚至一阶导数很小的局部峰值也能导致二阶导数过零点.
  • 为了避免噪声的影响,必须采用特别有效的滤波方法.
  • 边缘的方向信息丢失。

三:高斯-拉普拉斯算子(Log)

1.相关概念

        Marr和Hildreth将高斯滤波和拉普拉斯边缘检测结合在一起,首先用高斯函数先进行低通滤波,然后利用拉普拉斯算子进行高通滤波并提取零交叉点,形成LoG(Laplacian of Gaussian)算法,也称之为拉普拉斯高斯算法. 

  

2.Log边缘检测

(1)基本特征

  • 平滑滤波器是高斯滤波器.
  • 采用拉普拉斯算子计算二阶导数.
  • 边缘检测判据是二阶导数零交叉点并对应一阶导数的较大峰值.
  • 使用线性内插方法在子像素分辨率水平上估计边缘的位置.  
  • 为什么使用高斯滤波器:
    • 逼近必须能够抑制噪声效应
    • 必须尽量精确地确定边缘的位置  
    • 但平滑去噪和边缘检测是一对矛盾,抑制噪声和边缘精确定位是无法同时得到满足的。应用高斯函数的一阶导数,在二者之间获得最佳的平衡。

(2) 代码实战

对图像加入不同噪声之后使用Log算子进行边缘检测:

  1. I = rgb2gray(imread('image\room.tif'));
  2. J=imnoise(f,'gaussian',0.1);
  3. BW1 = edge(I,'log',0.004); % σ=2
  4. BW2 = edge(J,'log',0.012,2); % σ=2
  5. figure(1),subplot(121),imshow(J),title('gaussian = 0.1');
  6. subplot(122),imshow(BW2);title('THRESH = 0.012 σ=2');
  7. imshow(BW1);title('σ=2')
  8. BW1 = edge(I,'log',0.004,3); % σ=3
  9. figure, imshow(BW1);title('σ=3')

  

图3.4-1 Log算子(Sigma = 2)

  

图3.4-2 Log算子(Sigma = 3)

  

图3.4-3 Log算子(加入高斯0噪声,Sigma=2)

  

图3.4-4 Log算子(加入强度0.1高斯噪声,THRESH=0.01,Sigma=2)

  

图3.4-5 Log算子(加入强度0.1高斯噪声,THRESH=0.012,Sigma=2)

(3)结果分析

        Log算子我采用了matlab自带的函数进行实验,edge(I,'log',THRESH,SIGMA)

可以看到有两个参数,THRESH表示函数忽略所有强度不大于THRESH值的边缘,SIGMA作为LoG过滤器的标准差。

        LoG算子将高斯滤波和拉普拉斯边缘检测结合在一起,首先用高斯函数先进行低通滤波,然后利用拉普拉斯算子进行高通滤波并提取零交叉点。可以看到处理结果的图像边缘较平滑(见图3.4-1),是因为LoG算子不能检测边缘方向。而且在加入噪声之后,可以看到LoG算子对噪声十分敏感(见图3.4-3),不断调整参数,最终在参数THRESH=0.012,Sigma=2时得到较好的结果(见图3.4-5),但是还是损失了很多边缘信息。

四:Canny边缘检测

1.相概念

        Canny边缘检测是一种非常流行的边缘检测算法,计算机视觉中用得最广的边缘检测器,是John Canny在1986年提出的(发表在PAMI)。

        理论模型:加性噪声干扰的阶梯边缘(Step-Edge) Canny展示了高斯函数的一阶导紧密逼近最优化信噪比和局部化的边缘算子 

2.检测步骤

步骤1. 图像与高斯平滑滤波器卷积:

步2. 使用一阶有限差分计算偏导数阵列PQ

步骤3.  计算梯度幅值与方向角:

步骤4. 非极大值抑制(NMS ) :

    去掉幅值局部变化非极大的点.

    * 将梯度角离散为圆周的四个扇区之一,以便用3×3的窗口作抑制运算

   * 方向角离散化:

    

   * 抑制,得到新幅值图:

  

步骤5. 阈值化

取高低两个阈值作用于幅值图N[i,j],得到两个边缘图:高阈值和低阈值边缘图。连接高阈值边缘图,出现断点时,在低阈值边缘图中的8邻点域搜寻边缘点。

3.代码实战

  1. % I = rgb2gray(imread('image\room.tif'));
  2. % J=imnoise(I,'gaussian',0);
  3. % BW1 = edge(I,'canny',0.1);
  4. % BW2 = edge(J,'canny',0.175,1.9);
  5. % figure(2),subplot(121),imshow(J),title('加入gaussian噪声原图');
  6. % figure(2),subplot(122),imshow(BW2),title('加入gaussian噪声后检测结果');
  7. % figure(1),subplot(121),imshow(I),title('原图');
  8. % subplot(122),imshow(BW1);title('自带函数Canny边缘检测');
  9. % 自编程实现canny算子
  10. % image = rgb2gray(imread('image\lena.png'));
  11. % image=imnoise(image,'gaussian',0);
  12. image = imread('image\figure.png');
  13. image = rgb2gray(image);
  14. subplot(221);
  15. imshow(image);
  16. title('原始图像');
  17. image = double(image)/256;
  18. [m,n] = size(image);
  19. w = fspecial('gaussian');
  20. image_1 = imfilter(image,w,'replicate');
  21. subplot(222);
  22. imshow(int8(256*image_1));
  23. title('高斯滤波后的图像');
  24. % 梯度计算
  25. op = fspecial('sobel')/4; % 用Sobel算子来求导数
  26. x = op';
  27. y =op;
  28. bx = imfilter(image_1,x,'replicate');
  29. by = imfilter(image_1,y,'replicate');
  30. b_abs = sqrt(bx.*bx+by.*by); % 求梯度的幅值
  31. b_angle = angle(by-1i*bx);
  32. b_ang = b_angle/3.1416*180; % 求梯度的方向
  33. % 梯度方向确定
  34. dir=ones(m,n);
  35. for r = 1:m
  36. for c = 1:n
  37. if((b_ang(r,c)>=22.5 && b_ang(r,c)<67.5)|| (b_ang(r,c)>=-157.5 && b_ang(r,c)<-112.5))
  38. dir(r,c) = 1;
  39. elseif ((b_ang(r,c)>=67.5 && b_ang(r,c)<112.5)|| (b_ang(r,c)>=-112.5 && b_ang(r,c)<-67.5))
  40. dir(r,c) = 2;
  41. elseif ((b_ang(r,c)>=112.5 && b_ang(r,c)<157.5)|| (b_ang(r,c)>=-67.5 && b_ang(r,c)<-22.5))
  42. dir(r,c) = 3;
  43. else
  44. dir(r,c) = 0;
  45. end
  46. end
  47. end
  48. % 遍历图像(抑制非极大值)
  49. b_ab = [zeros(m,1),b_abs,zeros(m,1)]; % 串联矩阵
  50. b_ab = [zeros(1,n+2);b_ab;zeros(1,n+2)];
  51. b1=ones(m,n);
  52. for r = 2:m+1
  53. for c = 2:n+1
  54. switch dir(r-1,c-1)
  55. case 0
  56. if((b_ab(r,c)<b_ab(r+1,c))|| (b_ab(r,c)<b_ab(r-1,c)))
  57. b1(r-1,c-1) = 0;
  58. else
  59. b1(r-1,c-1) = b_ab(r,c);
  60. end
  61. case 1
  62. if((b_ab(r,c)<b_ab(r+1,c-1))|| (b_ab(r,c)<b_ab(r-1,c+1)))
  63. b1(r-1,c-1) = 0;
  64. else
  65. b1(r-1,c-1) = b_ab(r,c);
  66. end
  67. case 2
  68. if((b_ab(r,c)<b_ab(r,c-1))|| (b_ab(r,c)<b_ab(r,c+1)))
  69. b1(r-1,c-1) = 0;
  70. else
  71. b1(r-1,c-1) = b_ab(r,c);
  72. end
  73. case 3
  74. if((b_ab(r,c)<b_ab(r-1,c-1))|| (b_ab(r,c)<b_ab(r+1,c+1)))
  75. b1(r-1,c-1) = 0;
  76. else
  77. b1(r-1,c-1) = b_ab(r,c);
  78. end
  79. end
  80. end
  81. end
  82. % 高阈值保留
  83. b2=ones(m,n);
  84. for r = 1:m
  85. for c = 1:n
  86. if (b1(r,c)>0.5)
  87. b2(r,c) = 0;
  88. else
  89. b2(r,c) = 1;
  90. end
  91. end
  92. end
  93. % 低阈值舍弃
  94. b3=ones(m,n);
  95. for r = 1:m
  96. for c = 1:n
  97. if(b1(r,c)<0.08)
  98. b3(r,c) = 1;
  99. else
  100. b3(r,c) = 0;
  101. end
  102. end
  103. end
  104. % 介于高低阈值之间
  105. image_2 = b2;
  106. for r = 2:m-1
  107. for c = 2: n-1
  108. if (image_2(r,c)== 1 && (b3(r,c)==0||b3(r+1,c)==0 ...
  109. ||b3(r+1,c-1)==0||b3(r+1,c+1)==0||b3(r,c-1)==0 ...
  110. ||b3(r,c+1)==0||b3(r-1,c-1)==0||b3(r-1,c)==0 ...
  111. ||b3(r-1,c+1)==0))
  112. image_2(r,c) = 0;
  113. else
  114. image_2(r,c) = 1;
  115. end
  116. end
  117. end
  118. %去除黑边
  119. image_2(1,:) = 1;
  120. image_2(m,:) = 1;
  121. image_2(:,1) = 1;
  122. image_2(:,n) = 1;
  123. subplot(223);
  124. imshow(image_2);
  125. title('Canny算子检测后的图像');

4.结果分析 

  

图3.5-1 Matlab自带canny函数检测结果

图3.5-2 Matlab自带canny函数检测加入高斯噪声图像结果 

图3.5-3 自编程实现canny算子检测结果

  

图3.5-4 自编程实现canny算子检测加入高斯噪声图像结果

        Canny算子能够有效地抑制噪声,且其根据信噪比与定位比乘积进行测度,得到最优化逼近算子,能较精确确定边缘的位置。类似于LoG边缘检测方法,Canny也属于先求平滑后求导数的方法。Canny边缘检测算法可以简化为四个步骤:

步骤1:用高斯滤波器平滑处理原图像;
步骤2:用一阶偏导的有限差分进行计算梯度的幅值和方向;
步骤3:对梯度幅值进行非极大值抑制;
步骤4:用双阈值算法检测和连接边缘

        使用平滑滤波的原因从根本上来说是边缘检测算子中的导数计算。导数计算对噪声十分敏感,如果不提前使用滤波器加以改善,则在导数计算后,噪声将会被放大,使得检测出来的虚假边缘变多,不利于边缘的提取。但是平滑滤波虽然能有效抑制噪声,但是会使得图像边缘模糊,增加了边缘检测的不确定性,但是噪声带来的影响更大,因此首先应考虑去除噪声。图像的边缘有方向和幅度两个属性,沿边缘方向像素变化平缓,垂直于边缘方向像素变化剧烈,边缘上的这种变化可以用微分算子检测出来,通常用一阶或二阶导数来检测边缘。在这里选用一阶导数来检测边缘,即求一阶导数最大值。设M(x,y)代表幅值,当其取得局部最大值时,其对应的梯度方向θ(x,y)反映了边缘的方向(边缘方向与梯度方向垂直)。得到全局梯度之后,还不能确定边缘,还需要保留局部梯度最大的点,且还要抑制非极大值,即将非局部最大值的像素点置为0。这就需要利用梯度的方向,与领域像素的梯度幅值进行比较,在程序中我采用的是3*3的窗口做抑制运算,对每个像素点与沿着梯度方向的两个八领域像素的梯度进行比较,最后选取梯度值最大的点。这样就基本排除了非边缘信息,仅仅保留了一些细的线条,即候选边缘。最后还需要采用双阈值法来检测和连接边缘,将高于高阈值的像素点保留,将低于低阈值的像素点排除,若高阈值图像边缘出现0值像素点,则判断该像素点低阈值图像8领域空间的像素是否存在高于高阈值的像素,如果存在则该像素被保留(像素值置为1),否则被排除,至此就实现了Canny边缘检测。

        从实验结果(见图3.5-3)来看,Canny算子检测出来的边缘信息更准确,显示出来的边缘线条较细。且在加入噪声之后,由于前面利用平滑滤波过滤了噪声,最后得到的边缘检测结果也还是较满意的(见图3.5-4),但是由于原图像有些边缘周围的像素值较小,且变化较平缓,例如屋顶区域,在进行平滑滤波时容易将边缘信息抹除,故最终检测结果屋顶部分的边缘便消失了。

五:思考总结

1)评价一下Roberts 算子、Prewitt 算子、Sobel 算子对于噪声条件下边界检测的性能。

        从上面的检测结果来看,Roberts算子采用的是2*2的模板,对噪声比较敏感,对有噪声的图像处理能力不强;而Prewitt算子采用的是3*3的模板,其抗噪能力要强一些,但是效果也不是很理想;对于Sobel算子,其在Prewitt算子升增加了权重的概念,具有平滑噪声的作用,因此其对有噪声图像的边缘检测能力要比另外两个算子强。

2)为什么LoG梯度检测算子的处理结果不需要象Prewitt 等算子那样进行幅度组合?

        因为LoG算子采用拉普拉斯算子计算二阶导数,且边缘检测判据是二阶导数零交叉点并对应一阶导数的较大峰值,还使用线性内插方法在子像素分辨率水平上估计边缘的位置。

3)实验中所使用的四种算子所得到的边界有什么异同?

        Roberts、Prewitt、Sobel算子得到的边缘信息较粗,定位都不是很准确,而LoG算子得到的边缘信息定位较准确,且得到的边缘线条较细。对于有噪声的图像,Roberts、Prewitt、LoG算子都对噪声比较敏感,处理能力一般,而Sobel算子能对噪声进行平滑,其对噪声图片的边缘检测要比另外三个好。

4)比较边缘提取中LOG、Canny算子的边缘提取效果。

        相同尺度下的LOG算子总是能比Canny算子检测出更多的细节,Canny算子能对真正比较显著地边缘给出检测。LOG算子容易受尺度的影响,不同尺度下的边缘点要用不同尺度的LOG算子检测,Canny 算子受尺度的影响不太明显,不同尺度下,边缘点的位置都有偏差,但几乎相同;从对噪声的敏感程度来看,LOG边缘检测子是采用二阶导数过零点的检测方法,故对噪声更敏感一些,因此从抑制噪声方面来讲,Canny边缘检测子不容易受到噪声的干扰,而相同尺度下LOG算子却容易受到噪声的干扰,抑制噪声的能力要弱一些。LOG算子对噪声的抑制能力随着尺度的增加而增加,相同尺度下的Canny算子比LOG算子的抗噪声能力强,而LOG算子比Canny算子的边缘点准确;对于弱边缘,由于Canny边缘检测子采用两种不同的阈值分别检测强边缘和弱边缘,并且当弱边缘和强边缘相连时,才将弱边缘包含在输出图像中,故而Canny算子更能检测出真正的弱边缘,但是Canny边缘检测子检测出的边缘的位置会有一定范围的误差;对于假边缘,LOG边缘检测子相对比较容易受到噪声干扰,会检测出更多的细节,也容易检测出一些由于噪声引起的假边缘,但是LOG边缘检测子对边缘位置的检测还是很准确的。

文章知识点与官方知识档案匹配,可进一步学习相关知识
OpenCV技能树首页概览13293 人正在系统学习中