计算机视觉--Harris角点检测实现与分析(一)

一、Harris角点检测

1.1 何为角点?

      对于角点,到目前为止还没有明确的数学定义。但是我们可以认为角点就是极值点,即在某方面属性特别突出的点。一般的角点检测都是对有具体定义的、或者是能够具体检测出来的兴趣点的检测。这意味着兴趣点可以是角点,是在某些属性上强度最大或者最小的孤立点、线段的终点,或者是曲线上局部曲率最大的点。
      通俗的来说,在一副图像中,我们可以认为角点是物体轮廓线的连接点,当拍摄视角变化的时候,这些特征点仍能很好地保持稳定的属性。下面有两幅不同视角的图像,通过找出对应的角点进行匹配。
在这里插入图片描述
我们可以直观的概括下角点所具有的特征

  • 轮廓之间的交点;
  • 对于同一场景,即使视角发生变化,通常具备稳定性质的特征;
  • 该点附近区域的像素点无论在梯度方向上还是其梯度幅值上有着较大变化;

1.2 角点检测算法基本思想是什么?

      算法基本思想是使用一个固定窗口在图像上进行任意方向上的滑动,比较滑动前与滑动后两种情况,窗口中的像素灰度变化程度,如果存在任意方向上的滑动,都有着较大灰度变化,那么我们可以认为该窗口中存在角点。

1.3 如何用数学方法去刻画角点特征?

当窗口发生[u,v]移动时,那么滑动前与滑动后对应的窗口中的像素点灰度变化描述如下:
在这里插入图片描述
公式解释:

  • [u,v]是窗口的偏移量
  • (x,y)是窗口内所对应的像素坐标位置,窗口有多大,就有多少个位置
  • w(x,y)是窗口函数,最简单情形就是窗口内的所有像素所对应的w权重系数均为1。但有时候,我们会将w(x,y)函数设定为以窗口中心为原点的二元正态分布。如果窗口中心点是角点时,移动前与移动后,该点的灰度变化应该最为剧烈,所以该点权重系数可以设定大些,表示窗口移动时,该点在灰度变化贡献较大;而离窗口中心(角点)较远的点,这些点的灰度变化几近平缓,这些点的权重系数,可以设定小点,以示该点对灰度变化贡献较小,那么就想到使用二元高斯函数来表示窗口函数。

所以通常窗口函数有如下两种形式:
在这里插入图片描述

二、代码实现

实现Harris角点检测主要是通过调用PCV包中自带的harris.py文件中的函数。

# -*- coding: utf-8 -*-
from pylab import *
from PIL import Image
from PCV.localdescriptors import harris

"""
Example of detecting Harris corner points (Figure 2-1 in the book).
"""

# 读入图像
im = array(Image.open('../data/empire.jpg').convert('L'))

# 检测harris角点
harrisim = harris.compute_harris_response(im)

# Harris响应函数
harrisim1 = 255 - harrisim

figure()
gray()

#画出Harris响应图
subplot(141)
imshow(harrisim1)
print harrisim1.shape
axis('off')
axis('equal')

threshold = [0.01, 0.05, 0.1]
for i, thres in enumerate(threshold):
    filtered_coords = harris.get_harris_points(harrisim, 6, thres)
    subplot(1, 4, i+2)
    imshow(im)
    print im.shape
    plot([p[1] for p in filtered_coords], [p[0] for p in filtered_coords], '*')
    axis('off')

#原书采用的PCV中PCV harris模块
#harris.plot_harris_points(im, filtered_coords)

# plot only 200 strongest
# harris.plot_harris_points(im, filtered_coords[:200])

show()

三、不同场景下角点检测结果与分析

结果说明

以下每种场景,检测: 正面、侧面、旋转、尺度(远近)变化、光照变化的不同情况。每种情况下显示四组图,分别表示Harris响应图、阈值为0.01、 0.05以及0.1检测出的角点。

3.1 场景一:垂直或水平边缘多

3.1.1 检测结果

正面:
在这里插入图片描述
侧面:
在这里插入图片描述
光照较暗(傍晚拍摄):
在这里插入图片描述
正面远景:
在这里插入图片描述
旋转:
在这里插入图片描述

3.1.2 分析

  • 在垂直或水平边缘多的场景下,对于旋转变化、正侧面对比,基本不会对角点检测产生太大的影响,这是因为利用harris角点检测,对于垂直或水平边缘多的场景,边缘点附近会形成明显的梯度变化,因而可以明显地检测出角点。
  • 对于尺度变化(远近),可以看出远景下检测出的角点更多,主要因为扩大了范围。在远近不同的情况下,检测窗口尺寸不变的前提下,窗口内所包含图像的内容也是不同的,比较直观的说法即拍摄远景时,能看到更多的风景,因此检测出的角点自然会更加丰富,当然对比远近景下的同一座建筑物,远景中检测出的角点会因为建筑物在窗口中缩小导致检测出的角点变少。
  • 光线变化对比下并没有太大区别。说明 Harris角点检测算子对亮度和对比度的变化不敏感。这是因为在进行Harris角点检测时,使用了微分算子对图像进行微分运算,而微分运算对图像密度的拉升或收缩和对亮度的抬高或下降不敏感。换言之,对亮度和对比度的仿射变换并不改变Harris响应的极值点出现的位置,但是,由于阈值的选择,可能会影响角点检测的数量。
    在这里插入图片描述
  • 阈值越大,检测出的角点越少。

3.2 场景二:纹理角点丰富

3.2.1 检测结果

正面:
在这里插入图片描述
侧面:
在这里插入图片描述
远景:
在这里插入图片描述
正面旋转180度:
在这里插入图片描述
亮度变化(较暗):
在这里插入图片描述

3.2.2 分析

  • 对于纹理角点丰富的场景,在尺度变化(远近)时,在远景下,一些角点容易被遗漏,正面和侧面对比下,角点检测会出现较大差异,这是因为纹理角点丰富的场景,在Harris角点检测中,周围灰度值梯度较大的点较多,容易产生差异。
  • 旋转变化中,保持不变性,从Harris角点检测的原理分析,Harris角点检测算子使用的是角点附近的区域灰度值二阶矩阵。简写成如下形式:
    在这里插入图片描述
    用α,β表示矩阵M的特征值,二阶矩阵M可以看成是一个椭圆,那么α,β就是椭圆的长短轴的度量,如果仅仅是作了旋转变化,即椭圆转过一定角度,但是其形状保持不变(特征值保持不变),所以角点响应值R也不会发生变化。
  • 亮暗对比下,可以看出在较暗环境下深色物体(如以上结果图中的茶壶)几乎没有角点被检测出来,结合原理解释:深色物体本身灰度值与暗部的灰度值相近,因此不会产生边缘轮廓的灰度值差,所以难以检测。

3.3 场景三:平坦

3.3.1 检测结果

正面:
在这里插入图片描述
旋转:
在这里插入图片描述
侧面:
在这里插入图片描述
正面远景:
在这里插入图片描述
亮暗变化(较暗):
在这里插入图片描述

3.3.2 分析

  • 在纹理平坦的场景下,可以看出以上几种情况都能较清晰地检测出角点,这是因为平坦环境对角点检测干扰较小。
  • 在远近景对比下,会出现较大差异,远景中角点密集,猜测是目标物缩小,导致部分线段也会被检测成角点。说明Harris角点检测算子不具有尺度不变性。如下图所示,当右图被缩小时,在检测窗口尺寸不变的前提下,在窗口内所包含图像的内容是完全不同的。左侧的图像可能被检测为边缘或曲线,而右侧的图像则可能被检测为一个角点。
    在这里插入图片描述
  • 在较暗环境下,以上结果图中很清晰地检测出插座的四个角,原因是周围白色墙壁与暗处的插座灰度值形成更鲜明的对比,梯度突出;但是同理,在暗处插座孔与周围灰度值大致相同,因此很难被检测出角点。

四、角点检测响应函数R值的分析

写在另一篇博客中,点击查看: 计算机视觉–Harris角点检测实现与分析(二).

五、小结

实验结果小结:

  • 其他条件相同的情况下:阈值越小,找到的角点最多。
  • Harris角点检测具有旋转不变性。
  • Harris角点检测算子对亮度和对比度的变化不敏感。
  • Harris角点检测算子不具有尺度不变性。

本次实验过程中遇到的问题主要有:
需要输出图片时会出现如下报错
在这里插入图片描述
解决方案是:打开控制面板 - >系统设置 - >高级系统设置 - >环境变量->系统变量 - >新建。设置变量名称:QT_PLUGIN_PATH,变量目录:…\Anaconda2 \ Library \ plugins(电脑中的路径)。设置好环境变量后重启pycharm,问题即得到解决。

import cv2 as cv import numpy as np """"" cv2.cornerHarris() 可以用来进行角点检测。参数如下: • img - 数据类型为 float32 的输入图像。 • blockSize - 角点检测中要考虑的领域大小。 • ksize - Sobel 求导中使用的窗口大小 • k - Harris 角点检测方程中的自由参数,取值参数为 [0,04,0.06] """"" src_inital = cv.imread("E:/opencv/picture/building.jpg") src = cv.cvtColor(src_inital,cv.COLOR_BGR2GRAY) src = np.float32(src) dst = cv.cornerHarris(src,3,3,0.04) #R值是由det(M)-K(trace(M))*(trace(M)),当该是角时,该所对应的R值就会很大,通过设置对R的阈值,就可以筛选得到角 #这里的dst就是R值构成的灰度图像,灰度图像坐标会与原图像对应,R值就是角分数,当R值很大的时候 就可以认为这个是一个角 print(dst.shape) src_inital[dst>0.08*dst.max()]=[0,0,255] """"" src_inital[dst>0.08*dst.max()]=[0,0,255] 这句话来分析一下 dst>0.08*dst.max()这么多返回是满足条件的dst索引值,根据索引值来设置这个的颜色 这里是设定一个阈值 当大于这个阈值分数的都可以判定为角 dst其实就是一个个角度分数R组成的,当λ1和λ2都很大,R 也很大,(λ1和λ2中的最小值都大于阈值)说明这个区域是角。 那么这里为什么要大于0.08×dst.max()呢 注意了这里R是一个很大的值,我们选取里面最大的R,然后只要dst里面的值大于百分之八的R的最大值  那么此时这个dst的R值也是很大的 可以判定他为角,也不一定要0.08可以根据图像自己选取不过如果太小的话 可能会多圈出几个不同的角 """"" cv.imshow("inital_window",src_inital) cv.waitKey(0) cv.destroyAllWindows() 目标: 理解Harris角点检测的概念 使用函数cv2.cornerHarris(),cv2.cornerSubPix() 原理: Harris 角点检测的方法大概原理就是建立一个窗口区域,然后以当前窗口为中心向各个方向进行偏移。 如上图所示,第一个窗口向各个方向偏移的时候,像素值没有变化,因为窗口偏移的时候没有遇到任何边缘信息。 第二个图,窗口当中有一个直线(即block是在边缘上),如果当前窗口进行上下的移动,也没有像素值发生变化(在其他方向上灰度值也会变化)。 第三个图,窗口覆盖了一个“拐角”,如果窗口进行偏移,任何方向上都会有像素变化。 所以,第三张图片判断为检测到角。 判断特征是否为角的依据:R只与M值有关,R为大数值正数时特征为角,R为大数值负数时为边缘,R为小数值时为平坦区 寻找R位于一定阈值之上的局部最大值,去除伪角。 方向导数IxIx和IyIy可以使用cv2.Sobel()函数得到 Harris角点检测的结果是灰度图,图中的值为角点检测的打分值。需要选取合适的阈值对结果进行二值化来检测角
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值