news 2026/4/16 12:36:13

OpenCVSharp:在实际应用中使用 KAZE 算法进行特征匹配

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
OpenCVSharp:在实际应用中使用 KAZE 算法进行特征匹配

概述

前面已经介绍过了OpenCV中的很多特征检测算法,但是好像还不太清楚具体是怎么使用的,今天以一个完整的例子介绍具体如何使用这些特征检测算法。

效果:

实践

读取两张图像:

// 读取图像 using var img1 = new Mat(FirstImagePath, ImreadModes.Color); using var img2 = new Mat(SecondImagePath, ImreadModes.Color);

检测关键点和计算描述符:

using var descriptors1 = new Mat(); usingvar descriptors2 = new Mat(); usingvar matcher = new BFMatcher(NormTypes.L2SQR); usingvar kaze = KAZE.Create(); // 检测关键点和计算描述符 kaze.DetectAndCompute(img1, null, outvar keypoints1, descriptors1); kaze.DetectAndCompute(img2, null, outvar keypoints2, descriptors2);

使用KNN匹配:

// 使用KNN匹配 DMatch[][] matches = matcher.KnnMatch(descriptors1, descriptors2, 2);

KNN(K-近邻)匹配是一种基于特征点的图像匹配算法,它通过计算两幅图像中特征点之间的距离,为每个特征点找到K个最相似的特征点作为候选匹配对,然后使用比率测试或其他筛选方法来消除错误匹配,最终实现图像间的特征点对应关系建立,广泛应用于图像拼接、目标识别和三维重建等计算机视觉任务中。

前面我们创建了一个暴力匹配器(Brute-Force Matcher)的实例,用于特征点匹配。NormTypes.L2SQR参数指定了距离计算方式为欧几里得距离的平方(L2距离的平方),这种距离度量适用于SIFT、SURF等特征描述符。

查看KnnMatch的函数签名:

public DMatch[][] KnnMatch(Mat queryDescriptors, Mat trainDescriptors, int k, Mat? mask = null, bool compactResult = false)

这个KnnMatch函数是OpenCV中用于执行K近邻特征匹配的核心方法,它为查询描述符集合中的每个描述符找到训练描述符集合中距离最近的k个匹配项。函数返回一个二维数组DMatch[][],其中每个子数组包含对应查询描述符的k个最佳匹配结果,按距离从小到大排序。

KnnMatch函数参数说明:

参数名

类型

说明

queryDescriptors

Mat

查询图像的特征描述符矩阵,每一行代表一个特征点的描述符

trainDescriptors

Mat

训练图像的特征描述符矩阵,用于与查询描述符进行匹配

k

int

为每个查询描述符要查找的最佳匹配数量

mask

Mat?

可选掩码矩阵,用于指定哪些描述符对参与匹配,null表示全部参与

compactResult

bool

决定是否压缩结果,false时结果数组大小与查询描述符行数相同,true时排除完全被掩码的查询描述符

得到的结果如下所示:

进行投票唯一性:

// 投票唯一性 VoteForUniqueness(matches, mask); int nonZero = Cv2.CountNonZero(mask); private static void VoteForUniqueness(DMatch[][] matches, Mat mask, float uniqnessThreshold = 0.80f) { var maskData = newbyte[matches.Length]; var maskHandle = GCHandle.Alloc(maskData, GCHandleType.Pinned); using (var m = Mat.FromPixelData(matches.Length, 1, MatType.CV_8U, maskHandle.AddrOfPinnedObject())) { mask.CopyTo(m); for (int i = 0; i < matches.Length; i++) { // 这也被称为NNDR最近邻距离比率 if ((matches[i][0].Distance / matches[i][1].Distance) <= uniqnessThreshold) maskData[i] = 255; else maskData[i] = 0; } m.CopyTo(mask); } maskHandle.Free(); }

这段代码实现了特征匹配中的唯一性投票机制,也称为NNDR(最近邻距离比率)测试,用于筛选出可靠的匹配对。对每个查询特征点的KNN匹配结果进行唯一性检验,通过比较最佳匹配与次佳匹配的距离比率来判断匹配的可靠性,默认阈值为0.80f,这是经验值,可根据应用场景调整。

进一步过滤KNN匹配结果:

// 投票大小和方向 nonZero = VoteForSizeAndOrientation(keypoints2, keypoints1, matches, mask, 1.5f, 20); static int VoteForSizeAndOrientation(KeyPoint[] modelKeyPoints, KeyPoint[] observedKeyPoints, DMatch[][] matches, Mat mask, float scaleIncrement, int rotationBins) { int idx = 0; int nonZeroCount = 0; byte[] maskMat = newbyte[mask.Rows]; GCHandle maskHandle = GCHandle.Alloc(maskMat, GCHandleType.Pinned); using (Mat m = Mat.FromPixelData(mask.Rows, 1, MatType.CV_8U, maskHandle.AddrOfPinnedObject())) { mask.CopyTo(m); List<float> logScale = new List<float>(); List<float> rotations = new List<float>(); double s, maxS, minS, r; maxS = -1.0e-10f; minS = 1.0e10f; // 如果在这里得到异常,那是因为你传入的模型和观察关键点顺序错误。只需切换顺序。 for (int i = 0; i < maskMat.Length; i++) { if (maskMat[i] > 0) { KeyPoint observedKeyPoint = observedKeyPoints[i]; KeyPoint modelKeyPoint = modelKeyPoints[matches[i][0].TrainIdx]; s = Math.Log10(observedKeyPoint.Size / modelKeyPoint.Size); logScale.Add((float)s); maxS = s > maxS ? s : maxS; minS = s < minS ? s : minS; r = observedKeyPoint.Angle - modelKeyPoint.Angle; r = r < 0.0f ? r + 360.0f : r; rotations.Add((float)r); } } int scaleBinSize = (int)Math.Ceiling((maxS - minS) / Math.Log10(scaleIncrement)); if (scaleBinSize < 2) scaleBinSize = 2; float[] scaleRanges = { (float)minS, (float)(minS + scaleBinSize + Math.Log10(scaleIncrement)) }; usingvar scalesMat = Mat.FromArray(logScale.ToArray()); usingvar rotationsMat = Mat.FromArray(rotations.ToArray()); usingvar flagsMat = new Mat<float>(logScale.Count, 1); using Mat hist = new Mat(); flagsMat.SetTo(new Scalar(0.0f)); float[] flagsMatFloat1 = flagsMat.ToArray(); int[] histSize = { scaleBinSize, rotationBins }; float[] rotationRanges = { 0.0f, 360.0f }; int[] channels = { 0, 1 }; Rangef[] ranges = { new Rangef(scaleRanges[0], scaleRanges[1]), new Rangef(rotations.Min(), rotations.Max()) }; Mat[] arrs = { scalesMat, rotationsMat }; Cv2.CalcHist(arrs, channels, null, hist, 2, histSize, ranges); Cv2.MinMaxLoc(hist, outdouble minVal, outdouble maxVal); Cv2.Threshold(hist, hist, maxVal * 0.5, 0, ThresholdTypes.Tozero); Cv2.CalcBackProject(arrs, channels, hist, flagsMat, ranges); MatIndexer<float> flagsMatIndexer = flagsMat.GetIndexer(); for (int i = 0; i < maskMat.Length; i++) { if (maskMat[i] > 0) { if (flagsMatIndexer[idx++] != 0.0f) { nonZeroCount++; } else maskMat[i] = 0; } } m.CopyTo(mask); } maskHandle.Free(); return nonZeroCount; }

通过分析匹配对之间的尺度变化和旋转角度一致性来评估匹配质量。

获取匹配点:

// 收集好的匹配点 List<Point2f> obj = new List<Point2f>(); List<Point2f> scene = new List<Point2f>(); List<DMatch> goodMatchesList = new List<DMatch>(); // 遍历掩码,只提取非零项,因为它们是匹配项 for (int i = 0; i < mask.Rows; i++) { MatIndexer<byte> maskIndexer = mask.GetGenericIndexer<byte>(); if (maskIndexer[i] > 0) { obj.Add(keypoints1[matches[i][0].QueryIdx].Pt); scene.Add(keypoints2[matches[i][0].TrainIdx].Pt); goodMatchesList.Add(matches[i][0]); } } // 转换点类型 List<Point2d> objPts = obj.ConvertAll(Point2fToPoint2d); List<Point2d> scenePts = scene.ConvertAll(Point2fToPoint2d); private static Point2d Point2fToPoint2d(Point2f pf) { returnnew Point2d(((int)pf.X), ((int)pf.Y)); }

绘制结果图像:

// 计算单应性矩阵 Mat homography = Cv2.FindHomography(objPts, scenePts, HomographyMethods.Ransac, 1.5, mask); nonZero = Cv2.CountNonZero(mask); if (homography != null) { // 定义对象角点 Point2f[] objCorners = { new Point2f(0, 0), new Point2f(img1.Cols, 0), new Point2f(img1.Cols, img1.Rows), new Point2f(0, img1.Rows) }; // 透视变换 Point2d[] sceneCorners = MyPerspectiveTransform3(objCorners, homography); // 创建拼接图像 using Mat img3 = new Mat(Math.Max(img1.Height, img2.Height), img2.Width + img1.Width, MatType.CV_8UC3); using Mat left = new Mat(img3, new Rect(0, 0, img1.Width, img1.Height)); using Mat right = new Mat(img3, new Rect(img1.Width, 0, img2.Width, img2.Height)); img1.CopyTo(left); img2.CopyTo(right); // 获取掩码数组 mask.GetArray(outbyte[] maskBytes); // 绘制匹配 Cv2.DrawMatches(img1, keypoints1, img2, keypoints2, goodMatchesList, img3, Scalar.All(-1), Scalar.All(-1), maskBytes, DrawMatchesFlags.NotDrawSinglePoints); // 绘制检测到的对象边界 List<List<Point>> listOfListOfPoint2D = new List<List<Point>>(); List<Point> listOfPoint2D = new List<Point> { new Point(sceneCorners[0].X + img1.Cols, sceneCorners[0].Y), new Point(sceneCorners[1].X + img1.Cols, sceneCorners[1].Y), new Point(sceneCorners[2].X + img1.Cols, sceneCorners[2].Y), new Point(sceneCorners[3].X + img1.Cols, sceneCorners[3].Y) }; listOfListOfPoint2D.Add(listOfPoint2D); img3.Polylines(listOfListOfPoint2D, true, Scalar.LimeGreen, 2);
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/16 1:14:56

15、Windows 计算机管理与磁盘文件管理脚本详解

Windows 计算机管理与磁盘文件管理脚本详解 在 Windows 环境中,计算机管理和磁盘文件管理是常见且重要的任务。以下将详细介绍一些相关脚本的功能、使用方法及注意事项。 1. 修改页面文件大小 该脚本可用于修改多台计算机的页面文件大小。 - 脚本核心代码 : Dim cFil…

作者头像 李华
网站建设 2026/4/10 16:25:23

7、超融合架构:虚拟机器创建与高可用服务指南

超融合架构:虚拟机器创建与高可用服务指南 1. 虚拟机器高可用性概述 在超融合架构中,借助 Hyper - V 虚拟化技术,我们能够创建极其强大的故障转移解决方案。每个物理主机可以容纳多个虚拟机,这些虚拟机可以作为故障转移集群的成员节点。同时,物理主机本身也能成为故障转…

作者头像 李华
网站建设 2026/4/10 2:43:49

12、构建动态数据中心的关键技术与策略

构建动态数据中心的关键技术与策略 在当今数字化时代,构建一个高效、灵活且安全的动态数据中心对于企业的发展至关重要。本文将深入探讨一些关键技术和策略,包括应用流式传输、隔离环境、硬件整合、软件迁移以及测试环境的创建。 应用流式传输 应用流式传输是一种将应用程…

作者头像 李华
网站建设 2026/4/15 3:15:53

Linly-Talker生成视频的HDR10支持现状与未来路线

Linly-Talker生成视频的HDR10支持现状与未来路线 在虚拟主播、AI讲师和智能客服日益普及的今天&#xff0c;用户对数字人生成内容的视觉质量要求已不再满足于“能看”&#xff0c;而是追求“专业级”的观感体验。尤其是在4K电视、HDR显示器和YouTube/Bilibili等平台纷纷支持高动…

作者头像 李华
网站建设 2026/4/16 12:16:43

通信系统仿真:信道编码与解码_(1).通信系统基础v1

通信系统基础 1. 通信系统概述 通信系统是用于在两个或多个实体之间传输信息的系统。通信系统的主要组成部分包括信源、编码器、信道、解码器和信宿。信源负责生成信息&#xff0c;编码器将信息转换为适合传输的格式&#xff0c;信道是信息传输的媒介&#xff0c;解码器将接收…

作者头像 李华