最简单的目标跟踪(模版匹配)

一、概述

目标跟踪是计算机视觉领域的一个重要分支。研究的人很多,近几年也出现了很多很多的算法。大家看看淋漓满目的paper就知道了。但在这里,我们也聚焦下比较简单的算法,看看它的优势在哪里。毕竟有时候简单就是一种美。

在这里我们一起来欣赏下“模板匹配”这个简单点的跟踪算法。它的思想很简单,我们把要跟踪的目标保存好,然后在每一帧来临的时候,我们在整个图像中寻找与这个目标最相似的,我们就相信这个就是目标了。那如何判断相似呢?就用到了一些相关性的东西了,这个在我之前的一篇博文里面介绍过,大家可以参考下:

模板匹配中差值的平方和(SSD)与互相关准则的关系

http://blog.csdn.net/zouxy09/article/details/8549743

然后为了适应目标的变化,我们就需要随时更新我们要跟踪的目标。换句话来说,在跟踪t帧的时候,也就是在第t帧寻找目标的时候,是与t-1帧中我们找到的目标来进行比较的。这样目标的外观变化就会及时的更新。这个就叫做在线跟踪方法。当然了,这个策略会导致跟踪漂移的问题,这就是近几年很多跟踪算法关注的重要问题之一了。

二、代码实现

我的代码是基于VS2010+ OpenCV2.4.2的。代码可以读入视频,也可以读摄像头,两者的选择只需要在代码中稍微修改即可。对于视频来说,运行会先显示第一帧,然后我们用鼠标框选要跟踪的目标,然后跟踪器开始跟踪每一帧。对摄像头来说,就会一直采集图像,然后我们用鼠标框选要跟踪的目标,接着跟踪器开始跟踪后面的每一帧。具体代码如下:

simpleTracker.cpp

[cpp] view plaincopy

  1. // Object tracking algorithm using matchTemplate
  2. // Author : zouxy
  3. // Date   : 2013-10-28
  4. // HomePage : http://blog.csdn.net/zouxy09
  5. // Email  : [email protected]
  6. #include <opencv2/opencv.hpp>
  7. using namespace cv;
  8. using namespace std;
  9. // Global variables
  10. Rect box;
  11. bool drawing_box = false;
  12. bool gotBB = false;
  13. // bounding box mouse callback
  14. void mouseHandler(int event, int x, int y, int flags, void *param){
  15. switch( event ){
  16. case CV_EVENT_MOUSEMOVE:
  17. if (drawing_box){
  18. box.width = x-box.x;
  19. box.height = y-box.y;
  20. }
  21. break;
  22. case CV_EVENT_LBUTTONDOWN:
  23. drawing_box = true;
  24. box = Rect( x, y, 0, 0 );
  25. break;
  26. case CV_EVENT_LBUTTONUP:
  27. drawing_box = false;
  28. if( box.width < 0 ){
  29. box.x += box.width;
  30. box.width *= -1;
  31. }
  32. if( box.height < 0 ){
  33. box.y += box.height;
  34. box.height *= -1;
  35. }
  36. gotBB = true;
  37. break;
  38. }
  39. }
  40. // tracker: get search patches around the last tracking box,
  41. // and find the most similar one
  42. void tracking(Mat frame, Mat &model, Rect &trackBox)
  43. {
  44. Mat gray;
  45. cvtColor(frame, gray, CV_RGB2GRAY);
  46. Rect searchWindow;
  47. searchWindow.width = trackBox.width * 3;
  48. searchWindow.height = trackBox.height * 3;
  49. searchWindow.x = trackBox.x + trackBox.width * 0.5 - searchWindow.width * 0.5;
  50. searchWindow.y = trackBox.y + trackBox.height * 0.5 - searchWindow.height * 0.5;
  51. searchWindow &= Rect(0, 0, frame.cols, frame.rows);
  52. Mat similarity;
  53. matchTemplate(gray(searchWindow), model, similarity, CV_TM_CCOEFF_NORMED);
  54. double mag_r;
  55. Point point;
  56. minMaxLoc(similarity, 0, &mag_r, 0, &point);
  57. trackBox.x = point.x + searchWindow.x;
  58. trackBox.y = point.y + searchWindow.y;
  59. model = gray(trackBox);
  60. }
  61. int main(int argc, char * argv[])
  62. {
  63. VideoCapture capture;
  64. capture.open("david.mpg");
  65. bool fromfile = true;
  66. //Init camera
  67. if (!capture.isOpened())
  68. {
  69. cout << "capture device failed to open!" << endl;
  70. return -1;
  71. }
  72. //Register mouse callback to draw the bounding box
  73. cvNamedWindow("Tracker", CV_WINDOW_AUTOSIZE);
  74. cvSetMouseCallback("Tracker", mouseHandler, NULL );
  75. Mat frame, model;
  76. capture >> frame;
  77. while(!gotBB)
  78. {
  79. if (!fromfile)
  80. capture >> frame;
  81. imshow("Tracker", frame);
  82. if (cvWaitKey(20) == ‘q‘)
  83. return 1;
  84. }
  85. //Remove callback
  86. cvSetMouseCallback("Tracker", NULL, NULL );
  87. Mat gray;
  88. cvtColor(frame, gray, CV_RGB2GRAY);
  89. model = gray(box);
  90. int frameCount = 0;
  91. while (1)
  92. {
  93. capture >> frame;
  94. if (frame.empty())
  95. return -1;
  96. double t = (double)cvGetTickCount();
  97. frameCount++;
  98. // tracking
  99. tracking(frame, model, box);
  100. // show
  101. stringstream buf;
  102. buf << frameCount;
  103. string num = buf.str();
  104. putText(frame, num, Point(20, 20), FONT_HERSHEY_SIMPLEX, 1, Scalar(0, 0, 255), 3);
  105. rectangle(frame, box, Scalar(0, 0, 255), 3);
  106. imshow("Tracker", frame);
  107. t = (double)cvGetTickCount() - t;
  108. cout << "cost time: " << t / ((double)cvGetTickFrequency()*1000.) << endl;
  109. if ( cvWaitKey(1) == 27 )
  110. break;
  111. }
  112. return 0;
  113. }

三、结果

       我们对在目标跟踪领域一个benchmark的视频-david来测试下代码的效果。如下图所以,每帧的帧号在右上角所示。这个视频的光照变化是挺大的,但是简单的模板匹配方法还是可以挺有效的进行跟踪的,而且速度很快,在这个视频中,只花费了1ms左右(耗时的长度与目标框的大小和机器的性能有关)。

时间: 2024-08-04 19:23:52

最简单的目标跟踪(模版匹配)的相关文章

模式识别之目标跟踪---最简单的目标跟踪方法--------模板匹配与相关系数法

前言 模板匹配和相关系数法是目标跟踪的经典方法,它的优点有很多:简单准确,适用面广,抗噪性好,而且计算速度快.缺点是不能适应剧烈光照变化和目标剧烈形变. 所谓模板匹配法,就是指在一帧图像内寻找目标模板的位置,和模板最像的地方就是目标了.只要把全图的所有子区域和目标模板比较一下,找到最像目标模板的子区域,它就是目标的位置.如何度量子区域和目标模板的相似程度呢?最简单的办法就是计算这二者的相关系数. 相关系数 相关系数(r)是一种数学距离,可以用来衡量两个向量的相似程度.它起源于余弦定理:cos(A

目标跟踪积累总结

在过去的一年半里尝试了一些简单的目标跟踪算法,现总结如下: 1.  采用霍夫变换提取圆轮廓,对目标进行跟踪. 2.  利用颜色直方图,选择合适的阈值,提取目标的颜色特征进行跟踪. 3.  采用Camshift算法对目标进行跟踪. 4.  采用ASMS算法对目标进行跟踪. 5.  将目标跟踪与控制系统结合,采用Camshift算法对高速运动目标进行跟踪. 视频显示:http://v.youku.com/v_show/id_XMzQ0MTcyODYwNA==.html?spm=a2hzp.82447

目标跟踪之模板匹配---简单的模板匹配

一.概述 目标跟踪是计算机视觉领域的一个重要分支.研究的人很多,近几年也出现了很多很多的算法.大家看看淋漓满目的paper就知道了.但在这里,我们也聚焦下比较简单的算法,看看它的优势在哪里.毕竟有时候简单就是一种美. 在这里我们一起来欣赏下“模板匹配”这个简单点的跟踪算法.它的思想很简单,我们把要跟踪的目标保存好,然后在每一帧来临的时候,我们在整个图像中寻找与这个目标最相似的,我们就相信这个就是目标了.那如何判断相似呢?就用到了一些相关性的东西了,这个在我之前的一篇博文里面介绍过,大家可以参考下

目标跟踪之光流法---光流法简单介绍

光流的概念是Gibson在1950年首先提出来的.它是空间运动物体在观察成像平面上的像素运动的瞬时速度,是利用图像序列中像素在时间域上的变化以及相邻帧之间的相关性来找到上一帧跟当前帧之间存在的对应关系,从而计算出相邻帧之间物体的运动信息的一种方法.一般而言,光流是由于场景中前景目标本身的移动.相机的运动,或者两者的共同运动所产生的.其计算方法可以分为三类: (1)基于区域或者基于特征的匹配方法: (2)基于频域的方法: (3)基于梯度的方法: 简单来说,光流是空间运动物体在观测成像平面上的像素运

单目标模版匹配

就是实现这样: 其中,能够根据模版运算,自动实时从图中找出相同模版的地方.从而能够计算出镜头的相对位移. 模板匹配的工作方式    模板匹配的工作方式跟直方图的反向投影基本一样,大致过程是这样的:通过在输入图像上滑动图像块对实际的图像块和输入图像进行匹配.    假设我们有一张100x100的输入图像,有一张10x10的模板图像,查找的过程是这样的:  (1)从输入图像的左上角(0,0)开始,切割一块(0,0)至(10,10)的临时图像:  (2)用临时图像和模板图像进行对比,对比结果记为c: 

KCF目标跟踪方法分析与总结

KCF目标跟踪方法分析与总结 correlation filter Kernelized correlation filter tracking 读"J. F. Henriques, R. Caseiro, P. Martins, J. Batista, 'High-speed tracking with kernelized correlation filters'" 笔记 KCF是一种鉴别式追踪方法,这类方法一般都是在追踪过程中训练一个目标检测器,使用目标检测器去检测下一帧预测位置

计算机视觉目标跟踪的算法分类

摘自百度百科............. (1)基于区域的跟踪算法 基于区域的跟踪算法基本思想是:将目标初始所在区域的图像块作为目标模板,将目标模板与候选图像中所有可能的位置进行相关匹配,匹配度最高的地方即为目标所在的位置.最常用的相关匹配准则是差的平方和准则,(Sum of Square Difference,SSD). 起初,基于区域的跟踪算法中所用到的目标模板是固定的,如 Lucas 等人提出 Lucas-Kanade 方法,该方法利用灰度图像的空间梯度信息寻找最佳匹配区域,确定目标位置.之

OpenCV2马拉松第13圈——模版匹配

收入囊中 在http://blog.csdn.net/abcd1992719g/article/details/25505315这里,我们已经学习了如何利用反向投影和meanshift算法来在图像中查找给定模版图片的位置.meanshift针对的是单张图像,在连续图像序列的跟踪中,camshift(Continuously Adaptive Mean-SHIFT)是一种著名的算法.但在这里,我们先不讨论camshift,而是先讨论最简单的模版匹配. 模版匹配算法 opencv normalize

C++ Primer 学习笔记_29_操作符重载与转换(4)--转换构造函数和类型转换运算符归纳、operator new 和 operator delete 实现一个简单内存泄漏跟踪器

C++ Primer 学习笔记_29_操作符重载与转换(4)--转换构造函数和类型转换运算符归纳.operator new 和 operator delete 实现一个简单内存泄漏跟踪器 一.转换构造函数 可以用单个实参来调用的构造函数定义从形参类型到该类型的一个隐式转换.如下: class Integral { public: Integral (int = 0); //转换构造函数 private: int real; }; Integral A = 1; //调用转换构造函数将1转换为In