Opencv (C++)系列学习---模板匹配

简介: Opencv (C++)系列学习---模板匹配



1.模板匹配的定义

       模板匹配就是在整个图像区域发现与给定子图像匹配的小块区域,该匹配方法并不是基于直方图,而是使用一个图像块在输入图像上进行“”滑动“”。(也就是在图像上按照模板大小一块一块比对)

2.API介绍

void cv::matchTemplate(
cv::InputArray image //需要匹配的图像
cv::InputArray temp  //模板图像
cv::OutArray  result //存储的计算得到的结果
int           method  //匹配的方法

对于该算子需要注意的有两点:

1.result是存储匹配的结果,对于它的定义为单通道的大小为(image.width-temp.width+1,image.height-temp.height+1),也就是原图的大小裁去temple模板的宽高,不过其定义rows和cols需要交换位置(正常图像Mat定义先rows后cols),同时数据类型可以是CV_8UC1或者CV_32FC1。具体定义如下:

Mat result(image.cols-temp.cols+1,image.rows-temp.rows+1,CV_32FC1)

2.匹配方法介绍

cv::TM_SQDIFF==0(方差匹配法)

cv::TM_SQDIFF_NORMED==1(归一方差匹配法)

cv::TM_CCORR==2(相关性匹配方法)

cv::TM_CCORR_NORMED==3(归一化的互相关匹配法)

cv::TM_CCOEFF==4(相关系数匹配法)

cv::TM_CCOEFF_NORMED==5(归一化相关系数匹配方法)

对于前两种方差匹配方法0和1,完全匹配后值为0,不匹配值很大(值越小,匹配效果越好)。

对于中间两种相关匹配方法2和3,完全匹配后值很大,不匹配时值很小,接近于0。(值越大,匹配效果越好)。

对于最后两种相关系数匹配方法4和5,完全匹配会得到1,完全误匹配会得到-1。(分值介于-1-1,值越大,匹配效果越好)。

3.寻找最优匹配位置(匹配后的配套操作)

1.cv::normalize 归一化,在同样的代码中选择不同的匹配方法,其result值的结果也不一致,通过归一化算子中cv::NORM_MINMAX方向可以将各点的匹配结果线性映射到0-1之间。

2.cv::minMaxLoc()最小和最大点查找,通过该算子可以将result中结果进行查找,找到分数最小最大值和其在result中的位置。

4.具体代码

#include<iostream>
#include<opencv2\opencv.hpp>
#include<opencv2\highgui\highgui.hpp>
#include<opencv2\imgproc\imgproc.hpp>
using namespace std;
using namespace cv;
#define WINDOW_NAME1 "【原始图片】"
#define WINDOW_NAME2 "【效果窗口】"
//定义全局变量
Mat g_srcImage, g_templateImage, g_resultImage;
int g_nMatchmethod;
int g_nMatTrackbarNum = 5;
//定义全局函数
void on_Matching(int, void*);
int main(int argc,char** argv)
{
  //【1】载入原图像和模块板
  g_srcImage = imread("E:\\进度\\11-16\\模板匹配\\1.jpg",1);
  g_templateImage = imread("E:\\进度\\11-16\\模板匹配\\2.jpg",1);
  //【2】创建窗口
  namedWindow(WINDOW_NAME1,WINDOW_AUTOSIZE);
  namedWindow(WINDOW_NAME2, WINDOW_AUTOSIZE);
  //【3】创建滑动条并进行初始化
  createTrackbar("方法",WINDOW_NAME1,&g_nMatchmethod,g_nMatTrackbarNum,on_Matching);
  on_Matching(0,0);
  waitKey(0);
  return 0;
}
//回调函数
void on_Matching(int, void*)
{
  //【1】 给局部变量初始化
  Mat srcImage;
  g_srcImage.copyTo(srcImage);
  //【2】初始化用于结果输出的矩阵
  int resultImage_cols = g_srcImage.cols - g_templateImage.cols + 1;
  int resultImage_rows = g_srcImage.rows - g_templateImage.rows + 1;
  g_resultImage.create(resultImage_cols,resultImage_rows,CV_32FC1);
  //【3】进行模板匹配
  matchTemplate(g_srcImage,g_templateImage,g_resultImage,g_nMatchmethod);
  normalize(g_resultImage,g_resultImage,0,1,NORM_MINMAX);
  //【4】通过函数minMaxLoc 定位最匹配的位置
  double minvalue, maxValue;
  Point minLocation, maxLocation, MatLocation;
  minMaxLoc(g_resultImage,&minvalue,&maxValue,&minLocation,&maxLocation);
  
  //【5】对于方法SQDIFF和SQDIFF_NORMED越小值有着更高的匹配结果,而其余的方法,数值越大匹配效果越好。
  if (g_nMatchmethod == TM_SQDIFF || g_nMatchmethod == TM_SQDIFF_NORMED)
  {
    MatLocation = minLocation;
  }
  else
  {
    MatLocation = maxLocation;
  }
  //【6】绘制出矩阵,并显示最终结果
  rectangle(srcImage, MatLocation, Point(MatLocation.x + g_templateImage.cols, MatLocation.y + g_templateImage.rows), Scalar(0, 0, 255), 2, 16);
  rectangle(g_resultImage, MatLocation, Point(MatLocation.x + g_templateImage.cols, MatLocation.y + g_templateImage.rows), Scalar(0, 0, 255), 2, 16);
  imshow(WINDOW_NAME1,srcImage);
  imshow(WINDOW_NAME2,g_resultImage);
}

运行结果如下图:


通过对匹配方法的各个结果对比发现,归一化的匹配方法(2和3)在大多数情况下都会有好的结果,特别是室外环境的图像。相关系数方法的匹配效果更好,但是计算时间代价高。在实际的应用中,尤其是通过摄像头进行自动部件的检测或者特征跟踪,应该尝试使用所有的方法,从中找到一个同时兼顾效率和准确率的方法。

相关文章
|
2月前
|
存储 算法 C++
C++ STL 初探:打开标准模板库的大门
C++ STL 初探:打开标准模板库的大门
112 10
|
5天前
|
算法 网络安全 区块链
2023/11/10学习记录-C/C++对称分组加密DES
本文介绍了对称分组加密的常见算法(如DES、3DES、AES和国密SM4)及其应用场景,包括文件和视频加密、比特币私钥加密、消息和配置项加密及SSL通信加密。文章还详细展示了如何使用异或实现一个简易的对称加密算法,并通过示例代码演示了DES算法在ECB和CBC模式下的加密和解密过程,以及如何封装DES实现CBC和ECB的PKCS7Padding分块填充。
25 4
2023/11/10学习记录-C/C++对称分组加密DES
|
27天前
|
安全 编译器 C++
【C++11】可变模板参数详解
本文详细介绍了C++11引入的可变模板参数,这是一种允许模板接受任意数量和类型参数的强大工具。文章从基本概念入手,讲解了可变模板参数的语法、参数包的展开方法,以及如何结合递归调用、折叠表达式等技术实现高效编程。通过具体示例,如打印任意数量参数、类型安全的`printf`替代方案等,展示了其在实际开发中的应用。最后,文章讨论了性能优化策略和常见问题,帮助读者更好地理解和使用这一高级C++特性。
42 4
|
28天前
|
算法 编译器 C++
【C++】模板详细讲解(含反向迭代器)
C++模板是泛型编程的核心,允许编写与类型无关的代码,提高代码复用性和灵活性。模板分为函数模板和类模板,支持隐式和显式实例化,以及特化(全特化和偏特化)。C++标准库广泛使用模板,如容器、迭代器、算法和函数对象等,以支持高效、灵活的编程。反向迭代器通过对正向迭代器的封装,实现了逆序遍历的功能。
34 3
|
2月前
|
编译器 C语言 C++
配置C++的学习环境
【10月更文挑战第18天】如果想要学习C++语言,那就需要配置必要的环境和相关的软件,才可以帮助自己更好的掌握语法知识。 一、本地环境设置 如果您想要设置 C++ 语言环境,您需要确保电脑上有以下两款可用的软件,文本编辑器和 C++ 编译器。 二、文本编辑器 通过编辑器创建的文件通常称为源文件,源文件包含程序源代码。 C++ 程序的源文件通常使用扩展名 .cpp、.cp 或 .c。 在开始编程之前,请确保您有一个文本编辑器,且有足够的经验来编写一个计算机程序,然后把它保存在一个文件中,编译并执行它。 Visual Studio Code:虽然它是一个通用的文本编辑器,但它有很多插
|
1月前
|
编译器 C++
【c++】模板详解(1)
本文介绍了C++中的模板概念,包括函数模板和类模板,强调了模板作为泛型编程基础的重要性。函数模板允许创建类型无关的函数,类模板则能根据不同的类型生成不同的类。文章通过具体示例详细解释了模板的定义、实例化及匹配原则,帮助读者理解模板机制,为学习STL打下基础。
31 0
|
2月前
|
编译器 程序员 C++
【C++打怪之路Lv7】-- 模板初阶
【C++打怪之路Lv7】-- 模板初阶
18 1
|
2月前
|
Java 编译器 C++
c++学习,和友元函数
本文讨论了C++中的友元函数、继承规则、运算符重载以及内存管理的重要性,并提到了指针在C++中的强大功能和使用时需要注意的问题。
27 1
|
2月前
|
编译器 C语言 C++
C++入门6——模板(泛型编程、函数模板、类模板)
C++入门6——模板(泛型编程、函数模板、类模板)
60 0
C++入门6——模板(泛型编程、函数模板、类模板)
|
2月前
|
算法 编译器 C++
【C++篇】领略模板编程的进阶之美:参数巧思与编译的智慧
【C++篇】领略模板编程的进阶之美:参数巧思与编译的智慧
91 2