Opencv模板匹配

使用OpenCV和C++来识别彩色图片中的特定物体,如黑桃♠,通常涉及几个步骤:预处理图像、特征提取、对象检测等。下面是一个基本的示例代码,演示如何使用OpenCV的模板匹配方法来识别图片中的黑桃♠。

函数原型

void matchTemplate(InputArray image, InputArray temp, OutputArray result, int method);

void matchTemplate(
    InputArray image,     // 输入的大图像
    InputArray templ,     // 输入的模板图像
    OutputArray result,   // 输出的匹配结果矩阵
    int method            // 匹配方法
);

参数说明:
image: 输入的大图像,它可以是灰度图像或者是彩色图像。
templ: 模板图像,它通常是较小的图像,我们希望在大图像中寻找这个模板的位置。
result: 匹配结果矩阵,输出的是一个二维矩阵,它的大小取决于输入图像和模板的大小,以及所选择的匹配方法。每个元素代表了相应位置的匹配程度。
method: 匹配的方法,有多种可选的方法,包括但不限于:
CV_TM_SQDIFF: 平方差匹配法
CV_TM_SQDIFF_NORMED: 归一化的平方差匹配法
CV_TM_CCORR: 互相关匹配法
CV_TM_CCORR_NORMED: 归一化的互相关匹配法
CV_TM_CCOEFF: 交叉系数匹配法
CV_TM_CCOEFF_NORMED: 归一化的交叉系数匹配法

函数作用:
这个函数通过遍历输入图像中的每一个可能的位置,将模板图像与该位置上的图像区域进行比较,计算出一个相似度值。最终,所有的相似度值会被存储在一个矩阵中,这个矩阵就是 result 参数。在不同的匹配方法下,这个矩阵的值有不同的意义。
例如,对于归一化的交叉系数匹配法 (CV_TM_CCOEFF_NORMED),匹配得分接近于 1 表示非常好的匹配;而对于平方差匹配法 (CV_TM_SQDIFF),得分接近于 0 表示很好的匹配。

使用示例: 如果你想找到一个模板图像在另一张大图像中的位置,你可以使用 matchTemplate 函数并随后分析 result 矩阵来确定最佳匹配位置。对于某些匹配方法,你需要找到矩阵中的最大值;而对于其他方法,则需要找到最小值。
void minMaxLoc(InputArray src, double* minVal, double* maxVal=0, Point* minLoc=0, Point* maxLoc=0, InputArray mask=noArray());
void minMaxLoc(const SparseMat& src, double* minVal, double* maxVal, int* minIdx=0, int* maxIdx=0);
minMaxLoc 是 OpenCV 库中的一个函数,用于查找数组中的全局最小值和最大值,同时也可以找出这些值的位置。此函数适用于多种类型的数组,包括稠密数组和稀疏数组。

对于稠密数组(如 Mat 或 Umat):
void minMaxLoc(
    InputArray src,       // 输入数组
    double* minVal,       // 输出指针,指向最小值
    double* maxVal = 0,   // 输出指针,指向最大值(可选)
    Point* minLoc = 0,    // 输出指针,指向最小值的位置(可选)
    Point* maxLoc = 0,    // 输出指针,指向最大值的位置(可选)
    InputArray mask = noArray() // 可选的掩码数组,用于指定要处理的像素子集
);


对于稀疏数组(如 SparseMat):
void minMaxLoc(
    const SparseMat& src, // 输入稀疏数组
    double* minVal,       // 输出指针,指向最小值
    double* maxVal,       // 输出指针,指向最大值
    int* minIdx = 0,      // 输出指针,指向最小值的索引(可选)
    int* maxIdx = 0       // 输出指针,指向最大值的索引(可选)
);


参数说明:
src: 输入的数组,可以是多通道的数组。如果是多通道的数组,函数会分别在每个通道上查找最小值和最大值。
minVal: 输出指针,指向最小值。如果不需要这个值,可以传入 nullptr。
maxVal: 输出指针,指向最大值。如果不需要这个值,可以传入 nullptr。
minLoc: 输出指针,指向最小值的位置。对于多通道数组,返回的是通道索引和像素坐标。如果不需要这个位置信息,可以传入 nullptr。
maxLoc: 输出指针,指向最大值的位置。对于多通道数组,返回的是通道索引和像素坐标。如果不需要这个位置信息,可以传入 nullptr。
mask: 可选的掩码数组,用于指定要处理的像素子集。掩码数组应该是单通道的 8 位无符号整型数组。非零值表示对应的像素应该被处理,零值则表示对应的像素被忽略。
rectangle 函数是 OpenCV 库中的一个函数,用于在图像上绘制矩形。这个函数有两种重载形式,一种接受两个顶点来定义矩形的对角线,另一种接受一个 Rect 对象来定义矩形。下面是这两个函数的中文含义及参数说明:

void rectangle(Mat& img, Point pt1, Point pt2, const Scalar& color, int thickness=1, int lineType=8, int shift=0);
void rectangle(Mat& img, Rect r, const Scalar& color, int thickness=1, int lineType=8, int shift=0);


对于使用两个顶点定义矩形的情况:
void rectangle(
    Mat& img,       // 要绘制矩形的图像
    Point pt1,      // 矩形左上角的点
    Point pt2,      // 矩形右下角的点
    const Scalar& color, // 矩形的颜色
    int thickness = 1,   // 线条的厚度,如果为负值,则填充矩形
    int lineType = 8,    // 线条类型,通常为 8 表示 8 连续性
    int shift = 0        // 坐标轴的位移量
);
对于使用 Rect 定义矩形的情况:
void rectangle(
    Mat& img,       // 要绘制矩形的图像
    Rect r,         // 矩形区域
    const Scalar& color, // 矩形的颜色
    int thickness = 1,   // 线条的厚度,如果为负值,则填充矩形
    int lineType = 8,    // 线条类型,通常为 8 表示 8 连续性
    int shift = 0        // 坐标轴的位移量
);

参数说明:
img: 输入/输出图像,矩形将在其上绘制。
pt1: 矩形左上角的坐标。
pt2: 矩形右下角的坐标。
r: 代表矩形的 Rect 对象。
color: 矩形的颜色,由 Scalar 类型定义。例如 Scalar(0, 0, 255) 表示红色。
thickness: 线条的厚度,单位为像素。默认值为 1。如果设置为负数,例如 -1,则矩形会被填充。
lineType: 线条类型,默认为 8,意味着使用 8 邻域的抗锯齿线条。其他可能的值有 LINE_4, LINE_8, LINE_AA。
shift: 坐标轴的位移量。通常情况下,设置为 0 即可。

//minMaxLoc 函数举例

#include <opencv2/opencv.hpp>
#include <iostream>

int main()
{
	cv::Mat image = cv::imread("10.jpg", cv::IMREAD_GRAYSCALE);

	if (image.empty())
	{
		std::cout << "Could not open or find the image" << std::endl;
		return -1;
	}

	double minVal, maxVal;
	cv::Point minLoc, maxLoc;

	cv::minMaxLoc(image, &minVal, &maxVal, &minLoc, &maxLoc);

	std::cout << "Minimum value: " << minVal << " at (" << minLoc.x << ", " << minLoc.y << ")" << std::endl;
	std::cout << "Maximum value: " << maxVal << " at (" << maxLoc.x << ", " << maxLoc.y << ")" << std::endl;

	system("pause");
	
	return 0;
}

原图

截取模板

代码

#include <iostream>
#include <opencv2/opencv.hpp>
#include <opencv2/imgproc.hpp>
#include <opencv2/highgui.hpp>

using namespace std;
using namespace cv;

int main()
{
	Mat srcImage = imread("10.jpg", IMREAD_COLOR);//读取源图像
	Mat templateImage = imread("10_0.jpg", IMREAD_COLOR);//读模板图像

	if (srcImage.empty() || templateImage.empty())
	{
		cout << "读入的图片有空资源!" << endl;
	}
	//转化为灰度图、以减少计算复杂度
	Mat graySrc, grayTemplate;
	cvtColor(srcImage, graySrc, COLOR_BGR2GRAY);
	cvtColor(templateImage, grayTemplate, COLOR_BGR2GRAY);

	//执行模板匹配
	Mat result;
	matchTemplate(graySrc, grayTemplate, result, TM_CCOEFF_NORMED);

	//设定阈值,找到匹配位置
	double minVal, maxVal;
	Point minLoc, maxLoc;
	minMaxLoc(result, &minVal, &maxVal, &minLoc, &maxLoc);

	//如果匹配得分足够高,则在原图上标出匹配区域
	int width = templateImage.cols;
	int height = templateImage.rows;
	if (maxVal > 0.8)//设置一个阈值,根据实际情况调整
	{
		rectangle(srcImage, maxLoc, Point(maxLoc.x + width, maxLoc.y + height), Scalar(0, 0, 255), 2);
	}
	//显示结果
	
	imshow("源图像", srcImage);
	imshow("模板图像", templateImage);

	waitKey(0);

	return 0;
}

运行结果

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值