OpenCV实现图像特征提取与匹配
一、特征检测与描述子提取
-
选择特征检测器
常用算法包括:- ORB:一种高效的替代SIFT和SURF的算法,主要用于移动机器人和增强现实等领域。适合实时应用,结合FAST关键点与BRIEF描述子。
- SIFT(尺度不变特征变换):一种用于图像特征检测与描述的经典算法,具有尺度、旋转、光照不变性等特点。计算成本较高,不适合实时应用 。(需OpenCV contrib模块)。
- SURF:是对SIFT的改进,主要用于实时应用中。(需OpenCV contrib模块)。
- BRISK:二进制描述子,速度快且对噪声鲁棒。主要用于实时应用中。
算法 特点 SIFT 尺度不变性:通过构建图像金字塔和高斯差分金字塔来检测关键点,具有较好的尺度不变性。
旋转不变性:为每个关键点分配一个基准方向,确保描述子的旋转不变性。
光照和视角变化鲁棒性:部分抵抗光照变化,对视角变化也有较好的适应性。
计算复杂度高:由于需要构建多尺度空间,计算成本较高,不适合实时应用。SURF 速度更快:通过使用Hessian矩阵近似替代SIFT中的高斯差分函数,提高了计算速度。
鲁棒性:在保持SIFT的优点基础上,进一步优化了特征点的检测和描述符的构建过程。
光照和尺度不变性:与SIFT类似,具有较好的光照和尺度不变性。ORB 速度快:基于FAST角点检测和BRIEF描述符,计算速度比SIFT和SURF快两个数量级。
旋转不变性:通过计算关键点的主方向,实现旋转不变性。
二进制描述符:使用二进制描述符,有利于加速特征匹配过程。BRISK 快速计算:通过使用圆形邻域和高效的采样模式,计算速度较快。
二进制描述符:使用二进制描述符,有利于加速特征匹配过程。
尺度不变性:通过构建多尺度空间,具有一定的尺度不变性。// ORB检测器初始化 cv::Ptr<cv::ORB> detector = cv::ORB::create(500); // 提取500个特征点 std::vector<cv::KeyPoint> keypoints1, keypoints2; cv::Mat descriptors1, descriptors2;// 检测关键点并计算描述子 detector->detectAndCompute(img1, cv::noArray(), keypoints1, descriptors1); detector->detectAndCompute(img2, cv::noArray(), keypoints2, descriptors2);// BRISK检测器 cv::Ptr<cv::BRISK> brisk = cv::BRISK::create(); // SURF检测器 cv::Ptr<cv::xfeatures2d::SURF> surf = cv::xfeatures2d::SURF::create(400);//SIFT// 创建 SIFT 检测器对象Ptr<SIFT> sift = SIFT::create();// 检测关键点并计算描述子std::vector<KeyPoint> keypoints;Mat descriptors;sift->detectAndCompute(img, noArray(), keypoints, descriptors);// 初始化SURF检测器,可以选择设置不同的阈值、半径等参数double hessianThreshold = 400;int nOctaves = 4;int nOctaveLayers = 3;bool extended = false;bool upright = false;cv::Ptr<cv::xfeatures2d::SURF> detector = cv::xfeatures2d::SURF::create(hessianThreshold, nOctaves, nOctaveLayers, extended, upright);// 检测关键点和提取特征描述子std::vector<cv::KeyPoint> keypoints1, keypoints2;cv::Mat descriptors1, descriptors2;detector->detectAndCompute(img1, cv::noArray(), keypoints1, descriptors1);detector->detectAndCompute(img2, cv::noArray(), keypoints2, descriptors2);
二、特征匹配
-
暴力匹配(BFMatcher)
计算所有特征对的距离,适合小规模数据集。cv::BFMatcher matcher(cv::NORM_HAMMING); // ORB/BRIEF用汉明距离 std::vector<cv::DMatch> matches; matcher.match(descriptors1, descriptors2, matches); -
FLANN匹配
使用近似最近邻搜索,适合大规模数据集。cv::FlannBasedMatcher matcher(new cv::flann::LshIndexParams(20, 10, 2)); matcher.match(descriptors1, descriptors2, matches);
三、匹配结果优化
-
过滤低质量匹配
通过距离阈值筛选:double min_dist = DBL_MAX, max_dist = 0; for (auto& m : matches) {if (m.distance < min_dist) min_dist = m.distance;if (m.distance > max_dist) max_dist = m.distance; } std::vector<cv::DMatch> good_matches; for (auto& m : matches) {if (m.distance < 3 * min_dist) // 经验阈值good_matches.push_back(m); } -
可视化匹配结果
cv::Mat img_matches; cv::drawMatches(img1, keypoints1, img2, keypoints2, good_matches, img_matches); cv::imshow("Matches", img_matches); cv::waitKey(0);
四、完整流程示例
4.1 ORB
#include <opencv2/opencv.hpp>
#include <opencv2/xfeatures2d.hpp>int main() {// 读取图像cv::Mat img1 = cv::imread("image1.jpg", cv::IMREAD_GRAYSCALE);cv::Mat img2 = cv::imread("image2.jpg", cv::IMREAD_GRAYSCALE);// 初始化ORB检测器cv::Ptr<cv::ORB> detector = cv::ORB::create(500);std::vector<cv::KeyPoint> kp1, kp2;cv::Mat desc1, desc2;detector->detectAndCompute(img1, cv::noArray(), kp1, desc1);detector->detectAndCompute(img2, cv::noArray(), kp2, desc2);// 暴力匹配cv::BFMatcher matcher(cv::NORM_HAMMING);std::vector<cv::DMatch> matches;matcher.match(desc1, desc2, matches);// 过滤匹配点double min_dist = 100;std::vector<cv::DMatch> good_matches;for (auto& m : matches) {if (m.distance < 2 * min_dist)good_matches.push_back(m);}// 可视化cv::Mat img_matches;cv::drawMatches(img1, kp1, img2, kp2, good_matches, img_matches);//可调整参数显示匹配连线颜色/粗细cv::imshow("Matches", img_matches);cv::waitKey(0);return 0;
}
关键参数说明
detector->create() 控制特征点数量(如ORB::create(500))
NORM_HAMMING 二进制描述子(ORB、BRISK)的距离计算方式
drawMatches() 可调整参数显示匹配连线颜色/粗细
4.2 SIFT
//SIFT特征提取与显示代码
#include <opencv2/opencv.hpp>
#include <opencv2/xfeatures2d.hpp> // SIFT 头文件using namespace cv;
using namespace cv::xfeatures2d;int main() {// 1. 读取图像Mat img = imread("image.jpg", IMREAD_GRAYSCALE);if (img.empty()) {std::cerr << "图像读取失败!" << std::endl;return -1;}// 2. 创建 SIFT 检测器对象Ptr<SIFT> sift = SIFT::create(); // 新版本接口// 3. 检测关键点并计算描述子std::vector<KeyPoint> keypoints;Mat descriptors;sift->detectAndCompute(img, noArray(), keypoints, descriptors); // 核心函数// 4. 绘制关键点(带尺度和方向)Mat img_keypoints;drawKeypoints(img, keypoints, img_keypoints, Scalar::all(-1), DrawMatchesFlags::DRAW_RICH_KEYPOINTS); // 可视化增强// 5. 显示结果imshow("SIFT 关键点", img_keypoints);waitKey(0);return 0;
}
//SIFT特征匹配代码
#include <opencv2/opencv.hpp>
#include <opencv2/xfeatures2d.hpp> // SIFT 头文件using namespace cv;
using namespace cv::xfeatures2d;int main()
{
// 1. 读取两幅图像并计算特征
Mat img1 = imread("image1.jpg", IMREAD_GRAYSCALE);
Mat img2 = imread("image2.jpg", IMREAD_GRAYSCALE);
std::vector<KeyPoint> kp1, kp2;
Mat desc1, desc2;
sift->detectAndCompute(img1, noArray(), kp1, desc1);
sift->detectAndCompute(img2, noArray(), kp2, desc2);// 2. 使用 FLANN 匹配器(需转换为 CV_32F)
desc1.convertTo(desc1, CV_32F);
desc2.convertTo(desc2, CV_32F);
FlannBasedMatcher matcher;
std::vector<DMatch> matches;
matcher.match(desc1, desc2, matches); // 快速近邻匹配// 3. 筛选优质匹配(阈值法)
std::sort(matches.begin(), matches.end());
const int keep = matches.size() * 0.15;
matches.erase(matches.begin() + keep, matches.end());// 4. 绘制匹配结果
Mat img_matches;
drawMatches(img1, kp1, img2, kp2, matches, img_matches);
imshow("特征匹配结果", img_matches);
waitKey(0);return 0;
}
关键参数说明
SIFT::create():创建 SIFT 检测器对象,可设置参数(如特征点数量、对比度阈值等)。
detectAndCompute():单函数完成关键点检测与描述子计算,提升效率。
drawKeypoints():DRAW_RICH_KEYPOINTS 参数绘制关键点的尺度圆和方向线。
4.3 SURF
//SURF特征描述子匹配代码
#include <opencv2/opencv.hpp>
#include <opencv2/xfeatures2d.hpp>int main() {// 加载图片cv::Mat img1 = cv::imread("path_to_image1.jpg", cv::IMREAD_GRAYSCALE);cv::Mat img2 = cv::imread("path_to_image2.jpg", cv::IMREAD_GRAYSCALE);// 初始化SURF检测器,可以选择设置不同的阈值、半径等参数double hessianThreshold = 400;int nOctaves = 4;int nOctaveLayers = 3;bool extended = false;bool upright = false;cv::Ptr<cv::xfeatures2d::SURF> detector = cv::xfeatures2d::SURF::create(hessianThreshold, nOctaves, nOctaveLayers, extended, upright);// 检测关键点和提取特征描述子std::vector<cv::KeyPoint> keypoints1, keypoints2;cv::Mat descriptors1, descriptors2;detector->detectAndCompute(img1, cv::noArray(), keypoints1, descriptors1);detector->detectAndCompute(img2, cv::noArray(), keypoints2, descriptors2);// 使用FLANN匹配器进行特征匹配cv::FlannBasedMatcher matcher(cv::makePtr<cv::flann::LshIndexParams>(12, 20, 2));std::vector<cv::DMatch> matches;matcher.match(descriptors1, descriptors2, matches);// 根据匹配距离进行排序std::sort(matches.begin(), matches.end());// 绘制匹配结果cv::Mat matchesImage;cv::drawMatches(img1, keypoints1, img2, keypoints2, matches, matchesImage,cv::Scalar::all(-1), cv::Scalar::all(-1), std::vector<char>(), cv::DrawMatchesFlags::NOT_DRAW_SINGLE_POINTS);// 显示匹配结果图像cv::imshow("Matches", matchesImage);cv::waitKey(0);return 0;
}
4.4 BRISK
#include <opencv2/opencv.hpp>
#include <opencv2/xfeatures2d.hpp> // 需要 OpenCV contrib 模块using namespace cv;int main() {// 1. 读取图像Mat img1 = imread("image1.jpg", IMREAD_GRAYSCALE);Mat img2 = imread("image2.jpg", IMREAD_GRAYSCALE);// 2. 初始化 BRISK 检测器Ptr<BRISK> brisk = BRISK::create();std::vector<KeyPoint> keypoints1, keypoints2;Mat descriptors1, descriptors2;// 3. 提取关键点和描述子brisk->detectAndCompute(img1, noArray(), keypoints1, descriptors1);brisk->detectAndCompute(img2, noArray(), keypoints2, descriptors2);// 4. 使用暴力匹配器(汉明距离)BFMatcher matcher(NORM_HAMMING);std::vector<DMatch> matches;matcher.match(descriptors1, descriptors2, matches);// 5. 过滤匹配点(基于距离阈值)double min_dist = 100;for (const auto& m : matches) {if (m.distance < min_dist) min_dist = m.distance;}std::vector<DMatch> good_matches;for (const auto& m : matches) {if (m.distance < 2 * min_dist) { // 调整阈值以控制筛选严格度good_matches.push_back(m);}}// 6. 可视化关键点Mat img_kp1, img_kp2;drawKeypoints(img1, keypoints1, img_kp1, Scalar(0, 255, 0), DrawMatchesFlags::DRAW_RICH_KEYPOINTS);drawKeypoints(img2, keypoints2, img_kp2, Scalar(0, 255, 0), DrawMatchesFlags::DRAW_RICH_KEYPOINTS);imshow("BRISK Keypoints 1", img_kp1);imshow("BRISK Keypoints 2", img_kp2);// 7. 可视化匹配结果Mat img_matches;drawMatches(img1, keypoints1, img2, keypoints2, good_matches, img_matches,Scalar::all(-1), Scalar::all(-1), std::vector<char>(), DrawMatchesFlags::NOT_DRAW_SINGLE_POINTS);imshow("BRISK Matches", img_matches);waitKey(0);return 0;
}
关键参数说明
1)BRISK 参数:
Ptr<BRISK> brisk = BRISK::create(
int thresh = 30, // FAST角点检测阈值
int octaves = 3, // 图像金字塔层数
float patternScale = 1.0 // 描述子采样模式缩放
);
修改 thresh 可调整检测到的关键点数量(值越小,检测到的点越多)。
2)匹配筛选:
调整 if (m.distance < 2 * min_dist) 中的倍数系数(如 2 → 3)可放宽或收紧筛选条件。
使用 RANSAC 几何验证 进一步优化匹配(参考 SIFT/SURF 代码中的 findHomography 步骤)。
相关文章:
OpenCV实现图像特征提取与匹配
一、特征检测与描述子提取 选择特征检测器 常用算法包括: ORB:一种高效的替代SIFT和SURF的算法,主要用于移动机器人和增强现实等领域。适合实时应用,结合FAST关键点与BRIEF描述子。SIFT(尺度不变特征变…...
将分支`XXX`合并到远程分支`master
将分支feat-task合并到远程分支master 首先,切换到本地的 master 分支 git checkout master确保你的本地 master 分支是最新的,拉取远程的更新 git pull origin master将 feat-task 分支的代码合并到 master 分支 git merge feat-task如果在合并过程…...
程序化广告行业(13/89):DSP的深入解析与运营要点
程序化广告行业(13/89):DSP的深入解析与运营要点 大家好!一直以来,我都对程序化广告行业保持着浓厚的学习兴趣,在探索的过程中积累了不少心得。今天就想把这些知识分享出来,和大家一起学习进步…...
XML文件格式的简介及如何用Python3处理XML格式对象
诸神缄默不语-个人技术博文与视频目录 文章目录 1. XML格式简介2. 格式化XML文件的工具3. Python处理XML:xml库1. xml.etree.\(c\)ElementTree2. xml.dom.minidom 4. 本文撰写过程中参考的其他网络资料 1. XML格式简介 可扩展标记语言 (Extensible Markup Language…...
通过qemu仿真树莓派系统调试IoT固件和程序
通过qemu仿真树莓派系统调试IoT固件和程序 本文将介绍如何使用 QEMU 模拟器在 x86 架构的主机上运行 Raspberry Pi OS(树莓派操作系统)。我们将从下载镜像、提取内核和设备树文件,到启动模拟环境,并进行一些常见的操作࿰…...
Oracle底层原理解析
Oracle 解析 1、union \ union all \ Intersect \ Minus内部处理机制(优化) 当查询语句中的where子句中使用到or时,可以用union all来代替。因为使用or查询语句的时候,引起全表扫描,并走索引查询 特别:当…...
深度解读DeepSeek部署使用安全(48页PPT)(文末有下载方式)
深度解读DeepSeek:部署、使用与安全 详细资料请看本解读文章的最后内容。 引言 DeepSeek作为一款先进的人工智能模型,其部署、使用与安全性是用户最为关注的三大核心问题。本文将从本地化部署、使用方法与技巧、以及安全性三个方面,对Deep…...
【前端三剑客】万字总结JavaScript
一、初识JavaScript 1.1 JavaScript 的作用 表单动态校验(密码强度检测) ( JS 产生最初的目的 )网页特效服务端开发(Node.js)桌面程序(Electron)App(Cordova)控制硬件-物联网(Ruff)游戏开发(cocos2d-js) 1.2 HTML/CSS/JS 的关系…...
【哈希表与字符串的算法之路:思路与实现】—— LeetCode
文章目录 两数之和面试题01.02.判定是否为字符重排存在重复元素存在重复元素||字母异位词分组最长公共前缀和最长回文子串二进制求和字符串相乘 两数之和 这题的思路很简单,在读完题目之后,便可以想到暴力枚举,直接遍历整个数组两遍即可&…...
基于Android的记事本APP设计与实现:从需求分析到功能实现(超级简单记事本,附源码+文档报告)
基于Android的记事本APP设计与实现:从需求分析到功能实现 (以前大学课堂作业,抄在这里当个回忆吧) 引言 随着社会的不断进步,信息化建设不断发展,电子文字输入在生活、学习、工作中占有越来越重要的作用…...
eNSP中路由器的CON/AUX接口、GE Combo接口、Mini USB接口、USB接口、WAN侧uplink接口、FE接口、GE接口介绍
路由器常见接口的详细介绍及其应用示例: 1. CON/AUX 接口 全称:Console/Auxiliary(控制台/辅助接口)作用: CON(Console):通过命令行界面(CLI)直接配置路由器…...
Hello Mr. My Yesterday日文歌词附假名注音,祭奠逝去的青春
hello mr. my yesterday Hundred Percent Free Hello Mr. my yesterday云っておくれよ “夢叶うその瞬間にまた逢える”と 前方の幾多前途多難の未知 後方の道後悔も知った 経験と価値 夢なかば 一本の道結果だが ひとつだけ知りたいよ 神様がいるのなら “幸せの定義っ…...
ubuntu ollama+dify实践
安装ollama 官网的指令太慢了,使用以下指令加速: export OLLAMA_MIRROR"https://ghproxy.cn/https://github.com/ollama/ollama/releases/latest/download" curl -fsSL https://ollama.com/install.sh | sed "s|https://ollama.com/dow…...
S7-1200 G2移植旧版本S7-1200程序的具体方法示例
S7-1200 G2移植旧版本S7-1200程序的具体方法示例 前期概要: S7-1200 G2必须基于TIA博途V20,之前的程序可通过移植的方式在新硬件上使用。 该移植工具可自动将TIA Portal 项目从 S7-1200 移植到更新的S7-1200 G2。 注意: 该插件支持在同一TIA Portal项目实例内将软件和/或硬…...
新办公室哪款空气净化器除甲醛效果好?高效除甲醛,提升效率
现代办公环境中,空气质量对员工的健康与工作效率产生着不可忽视的影响。尤其是新装修的办公室,往往因为空气中的甲醛浓度超标而导致一系列健康问题。因此,选择一款性能优越的除甲醛空气净化器就显得尤为重要。合适的空气净化器不仅可以有效过…...
塑造企业数字化形象:企业信息化UI界面设计的关键要素
引言 在数字化转型的大潮中,企业信息化系统的UI(用户界面)界面设计不仅是技术实现的最后一环,更是塑造企业数字化形象、提升用户体验、增强业务效率的重要手段。优秀的UI设计能够直观展现企业价值观,提升用户粘性&…...
大视频背景暗黑风格的wordpress企业主题免费下载
整体风格是黑色的,首页首屏大视频背景,动态效果非常好。向下滚动时,滚动的特效也不错。 原文 https://www.bixugao.com/wp/26.html...
CUDA编程之内存零拷贝技术
一、实现原理 零拷贝内存通过将主机锁页内存直接映射到设备地址空间,实现CPU与GPU共享内存,避免显式数据拷贝。锁页内存通过cudaHostAlloc或cudaHostRegister分配,确保物理地址固定且不被操作系统换页,从而支持DMAÿ…...
C语言基础知识04
指针 指针概念 指针保存地址,地址是字节的编号 指针类型和保存的地址类型要一直 使用时注意,把地址转换为&变量的格式来看 int a[3]; a转为&a[0] 指针的大小 64bit 固定8字节, 32bit 固定4字节 指针…...
在 Java 中,== 和 equals 的区别
1. 运算符 作用:比较两个对象的 内存地址(引用类型)或 值(基本数据类型)。 适用场景: 基本数据类型(int, char, boolean 等):直接比较值是否相等。 引用类型ÿ…...
Qt开发:QtWebEngine中操作选择文本
查找选择 在QtWebEngine中,可以使用QWebEnginePage的findText方法来查找文本,查找成功以后,将自动选择当前文本。 QWebEnginePage可以通过QWebEngineView的page()来取得。 比如,如下代码可以在页面中查找hello,world并选择。 …...
VUE的脚手架搭建引入类库
VUE的小白脚手架搭建 真的好久好久自己没有发布自己博客了,对于一直在做后端开发的我 ,由于社会卷啊卷只好学习下怎么搭建前端,一起学习成长吧~哈哈哈(最终目的,能够懂并简易开发) 文章目录 VUE的小白脚手架搭建1.下载node.js2.安装vue脚手架3.创建一个项目4.代码规范约束配置(…...
分布式系统日志排查综合场景
排查背景 在一个大型分布式电商系统中,用户反馈在进行商品结算时出现了报错。系统由多个子系统构成,包括商品管理系统、订单系统、支付系统等,各子系统分布在不同服务器上,且日志文件分散存储。 排查过程 确定当前位置并切换到可…...
android lmkd.rc 介绍
service service lmkd /system/bin/lmkdclass coreuser lmkdgroup lmkd system readproccapabilities DAC_OVERRIDE KILL IPC_LOCK SYS_NICE SYS_RESOURCEcriticalsocket lmkd seqpacketpasscred 0660 system systemtask_profiles ServiceCapacityLow属于核心服务组࿰…...
Android Studio执行Run操作报Couldn‘t terminate previous instance of app错误
步骤1、在项目根目录下build.gradle文件最后添加如下内容 //自定义任务名:assembleAndInstall tasks.register(assembleAndInstall, Exec.class, new Action<Exec>() {Overridevoid execute(Exec exec) {//设置自定义任务组名exec.setGroup(custom task)//当…...
Matlab 双线性插值(二维)
文章目录 一、简介二、实现代码三、实现效果参考资料一、简介 双线性插值是一种 二维插值方法,用于计算 栅格(Grid) 或 像素点 之间的插值值。它主要用于 图像缩放、旋转、变换 等操作,以在新像素位置估算灰度值或颜色值。 如上图所示,假设存在一个二维离散函数(如图像)…...
1700. 无法吃午餐的学生数量
无法吃午餐的学生数量 题目描述尝试做法推荐做法 题目描述 学校的自助午餐提供圆形和方形的三明治,分别用数字 0 和 1 表示。所有学生站在一个队列里,每个学生要么喜欢圆形的要么喜欢方形的。 餐厅里三明治的数量与学生的数量相同。所有三明治都放在一个…...
uv命令介绍(高性能Python包管理工具,旨在替代pip、pip-tools和virtualenv等传统工具)
文章目录 **主要功能**1. **快速安装和管理 Python 包**2. **生成和管理锁文件 (requirements.lock)**3. **创建虚拟环境**4. **与 poetry 兼容** **核心优势**1. **极快的速度**:基于 Rust 实现,利用多线程和缓存大幅加速依赖解析。2. **轻量且独立**&a…...
杨辉三角形(信息学奥赛一本通-2043)
【题目描述】 例5.11 打印杨辉三角形的前n(2≤n≤20)行。杨辉三角形如下图: 当n5时 1 1 1 1 2 1 1 3 3 1 1 4 6 4 1 输出: 1 1 1 1 2 1 1 3 3 1 1 4 6 4 1 【输入】 输入行数n。 【输出】 输出如题述三角形。n行&#…...
使用easyexcel实现单元格样式设置和下拉框设置
1.单元格样式设置 1.1实体类 public class DemoData {ExcelProperty("PK")private String name;ExcelProperty("年龄")private int age;// 必须提供无参构造方法public DemoData() {}public DemoData(String name, int age) {this.name name;this.age …...
