Android技术知识Android开发经验谈Android开发

计算机视觉 OpenCV Android | 特征检测与匹配之角

2019-02-08  本文已影响11人  凌川江雪
本文要点总结(俩算法的联系与区别)

Harris角点检测与Shi-Tomasi角点检测都是经典的角点特征提取算法,
但两者在API的使用上有出入(详见文中代码或GitHub项目);


引子


0 角点的定义与作用

基本特征检测一章中,学习了关于边缘检测的知识,
图像边缘中,有一些特殊的像素点值得我们特别关注,
那就是图像边缘的角点
这些角点更能反映出图像中对象的整体特征
基于角点周围的像素块生成特征描述子可以更好地表述图像特征数据

本文首先笔记如何提取图像的角点特征。

1 Harris角点检测

关于角点特征提取最经典的算法之一就是Harris角点检测

Harris角点检测基本原理是对图像求导,对每个像素点生成二阶梯度图像
只是在卷积核使用的时候需要使用高斯核
得到图像X与Y方向的二阶矩
基于它们就可以得到如下Hessian矩阵

求得最大两个特征值 λ1 与 λ2,可以得到如下 角点响应值R

其中,系数K常见的取值范围为0.02~0.04。

  • 每个像素点有自己的一个响应值R
    也即有自己的一对特征值 λ1 与 λ2;
  • 全局像素则有多个R值;

根据M计算可以得到特征值 λ1、λ2,它们的值与角点的关系如下图:

Harris角点检测的API:

使用Harris角点检测函数计算得到图像角点的演示代码如下:

private void harrisCornerDemo(Mat src, Mat dst) {
  // 定义阈值T//初始化各种Mat对象
  int threshold = 100;
  Mat gray = new Mat();
  Mat response = new Mat();
  Mat response_norm = new Mat();

  // 角点检测
  Imgproc.cvtColor(src, gray, Imgproc.COLOR_BGR2GRAY);//转灰度!!!!!!
  Imgproc.cornerHarris(gray, response, 2, 3, 0.04);
  Core.normalize(response, response_norm, 0, 255, Core.NORM_MINMAX, CvType.CV_32F);//归一化

  // 绘制角点
  dst.create(src.size(), src.type());
  src.copyTo(dst);
  float[] data = new float[1];//!!!!!!!!!!
  for(int j=0; j<response_norm.rows(); j++ )
  {
    for(int i=0; i<response_norm.cols(); i++ )
    {
      response_norm.get(j, i, data);
      if((int)data[0] > 100)
      {
        Imgproc.circle(dst, new Point(i, j), 5, new Scalar(0, 0, 255), 
2, 8, 0);
        Log.i("Harris Corner", "find corner point……");
      }
    }
  }
  gray.release();
  response.release();
}
  • response_norm归一化后的响应值Mat对象
  • data[0]是某个响应值;
    >100认为其是一个较大的响应值,
    响应值大于指定阈值T(这里是100),则对应的像素点被认为是角点;
  • float[] data = new float[1] //在这里,可能有人有疑问, 数组长度只有 1
    get()方法,第三个参数要求是数组,
    get多个像素时,传入一个多元素空数组,常规理解操作;
    但当只要get一个像素,则需创建一个只有一个元素的数组!而非变量

    这种接口设计思想
    一个方法(如get())接口即可实现包含一到多个数据元素的形式参数的传入;
    而没必要去准备/重载两个方法——
    一个用来接收包含单个数据元素的变量型形参
    另一个用来接收包含多个数据元素的数组型形参;没必要这样了;

    即无论是负责接收数据的形参是包含 单个数据元素 还是 多个数据元素 ,一律按 数组型形参 处理,把数据形参位定义成数组类型,接口统一设计,一套代码,一个接口即可,省时省力!

注意,阈值T与绘制检测得到的角点数目相关,
T值越大,被过滤的响应像素点越多,留下来的就越可能是角点,反之亦然。

本章完整代码在文末GitHub里边的Feature2dMainActivity.java文件中,后续对此不再说明。

2 Shi-Tomasi角点检测

还有一种经常使用的角点检测方法称为Shi-Tomasi角点检测
其与Harris角点检测类似,这种方法同样是基于梯度图像发展而来的,
它是1994年由两位作者Jianbo Shi与Carlo Tomasi一起提出来的,
他们当时所发表的论文名为<<Good Feature to Track>>,
这也是为什么在OpenCV中使用同名函数来表示Shi-Tomasi角点检测的原因。

Shi-Tomasi角点检测与Harris角点检测唯一(指的是方法逻辑,不包括API,API的输出还不同) 不同的地方在于计算角点响应R值时使用的是如下方法:

如果R大于指定阈值T,则对应的像素点被认为是角点;
假设λ1、λ2为坐标,
则对角点的描述就是当λ1、λ2都大于阈值T=λmin的右上角时
角点响应值满足要求的区域,
如下图:

相关的API如下:

每个像素点有自己的一个响应值R,去全局像素最大的R为Rmax;

minDistance最终返回的角点之间的最小距离,小于这个距离则的角点被丢弃。
 mask:默认全部为零。
 blockSize:计算矩阵M时需要的,常取值为3。
 useHarrisDetector:是否使用Harris角点检测,true表示使用,若为false则使用Shi-Tomasi角点检测。
 k:当使用Harris角点检测的时候才使用。

实现shi-tomasi角点检测的demo:

private void shiTomasicornerDemo(Mat src, Mat dst) {

  // 变量定义
  double k = 0.04;
  int blockSize = 3;
  double qualityLevel= 0.01;
  boolean useHarrisCorner = false;

  // 角点检测
  Mat gray = new Mat();
  Imgproc.cvtColor(src, gray, Imgproc.COLOR_BGR2GRAY);
  MatOfPoint corners = new MatOfPoint();
  Imgproc.goodFeaturesToTrack(gray, corners, 100, qualityLevel, 10, new Mat(), blockSize, useHarrisCorner, k);

  // 绘制角点
  dst.create(src.size(), src.type());
  src.copyTo(dst);
  Point[] points = corners.toArray();
  for(int i=0; i<points.length; i++) {
    Imgproc.circle(dst, points[i], 5, new Scalar(0, 0, 255), 2, 8, 0);
  }
  gray.release();
}

完整的代码可参考GitHub项目。


参考材料
上一篇 下一篇

猜你喜欢

热点阅读