OpenCV中常用特征提取算法(SURF、ORB、SIFT和AKAZE)用法示例(C++和Python)

OpenCV 中提供了多種常用的特征提取算法,廣泛應用于圖像匹配、拼接、SLAM、物體識別等任務。以下是 OpenCV 中幾個主流特征提取算法的 用法總結與代碼示例,涵蓋 C++ 和 Python 兩個版本。


常用特征提取算法列表

算法特點是否需額外模塊
SIFT(尺度不變特征)穩定性強、可旋轉縮放xfeatures2d 模塊
SURF(加速穩健特征)快速但專利保護xfeatures2d 模塊
ORB(Oriented FAST + BRIEF)免費、高速OpenCV 主模塊
AKAZE適用于非線性尺度空間OpenCV 主模塊
BRISK二進制描述子、適用于實時應用OpenCV 主模塊

示例一:ORB 特征提取與匹配

Python 版:

import cv2# 加載圖像
img1 = cv2.imread("image1.jpg", cv2.IMREAD_GRAYSCALE)
img2 = cv2.imread("image2.jpg", cv2.IMREAD_GRAYSCALE)# 創建 ORB 檢測器
orb = cv2.ORB_create()# 檢測關鍵點和計算描述子
kp1, des1 = orb.detectAndCompute(img1, None)
kp2, des2 = orb.detectAndCompute(img2, None)# 創建匹配器
bf = cv2.BFMatcher(cv2.NORM_HAMMING, crossCheck=True)
matches = bf.match(des1, des2)# 排序并繪制前 20 個匹配點
matches = sorted(matches, key=lambda x: x.distance)
matched_img = cv2.drawMatches(img1, kp1, img2, kp2, matches[:20], None, flags=2)cv2.imshow("ORB Matches", matched_img)
cv2.waitKey(0)
cv2.destroyAllWindows()

C++ 版:

#include <opencv2/opencv.hpp>
#include <opencv2/features2d.hpp>int main()
{cv::Mat img1 = cv::imread("image1.jpg", cv::IMREAD_GRAYSCALE);cv::Mat img2 = cv::imread("image2.jpg", cv::IMREAD_GRAYSCALE);cv::Ptr<cv::ORB> orb = cv::ORB::create();std::vector<cv::KeyPoint> kp1, kp2;cv::Mat des1, des2;orb->detectAndCompute(img1, cv::noArray(), kp1, des1);orb->detectAndCompute(img2, cv::noArray(), kp2, des2);cv::BFMatcher matcher(cv::NORM_HAMMING, true);std::vector<cv::DMatch> matches;matcher.match(des1, des2, matches);std::sort(matches.begin(), matches.end(),[](const cv::DMatch& m1, const cv::DMatch& m2) {return m1.distance < m2.distance;});cv::Mat img_matches;cv::drawMatches(img1, kp1, img2, kp2, matches, img_matches);cv::imshow("ORB Matches", img_matches);cv::waitKey(0);return 0;
}

示例二:SIFT 特征提取(需額外模塊)

Python 版:

import cv2sift = cv2.SIFT_create()
img = cv2.imread("image.jpg", cv2.IMREAD_GRAYSCALE)
kp, des = sift.detectAndCompute(img, None)img_kp = cv2.drawKeypoints(img, kp, None, flags=cv2.DRAW_MATCHES_FLAGS_DRAW_RICH_KEYPOINTS)
cv2.imshow("SIFT Features", img_kp)
cv2.waitKey(0)

注意:SIFT 在 OpenCV 4.x 中已開放為免費使用,需 pip install opencv-contrib-python


C++ 版:

#include <opencv2/opencv.hpp>
#include <opencv2/xfeatures2d.hpp>int main()
{cv::Mat img = cv::imread("image.jpg", cv::IMREAD_GRAYSCALE);cv::Ptr<cv::SIFT> sift = cv::SIFT::create();std::vector<cv::KeyPoint> keypoints;cv::Mat descriptors;sift->detectAndCompute(img, cv::noArray(), keypoints, descriptors);cv::Mat output;cv::drawKeypoints(img, keypoints, output, cv::Scalar::all(-1), cv::DrawMatchesFlags::DRAW_RICH_KEYPOINTS);cv::imshow("SIFT", output);cv::waitKey(0);return 0;
}

示例三:AKAZE 特征提取

Python 版:

import cv2akaze = cv2.AKAZE_create()
img = cv2.imread("image.jpg", cv2.IMREAD_GRAYSCALE)
kp, des = akaze.detectAndCompute(img, None)img_kp = cv2.drawKeypoints(img, kp, None, color=(0,255,0))
cv2.imshow("AKAZE", img_kp)
cv2.waitKey(0)

C++ 版:

#include <opencv2/opencv.hpp>int main()
{cv::Mat img = cv::imread("image.jpg", cv::IMREAD_GRAYSCALE);cv::Ptr<cv::AKAZE> akaze = cv::AKAZE::create();std::vector<cv::KeyPoint> keypoints;cv::Mat descriptors;akaze->detectAndCompute(img, cv::noArray(), keypoints, descriptors);cv::Mat output;cv::drawKeypoints(img, keypoints, output);cv::imshow("AKAZE", output);cv::waitKey(0);return 0;
}

示例四:SURF 特征提取

小提示

SURF 屬于專利算法(由 Hessian Matrix 加速構建特征),因此默認在 opencv-python不可用,需要使用 opencv-contrib 版本:

  • Python: pip install opencv-contrib-python
  • C++: 編譯 OpenCV 時需啟用 opencv_contrib,并啟用 xfeatures2d 模塊。

Python 示例

import cv2
import numpy as np# 讀取圖像
img = cv2.imread("image.jpg", cv2.IMREAD_GRAYSCALE)# 創建 SURF 對象(閾值越高,檢測特征點越少)
surf = cv2.xfeatures2d.SURF_create(hessianThreshold=400)# 檢測并計算特征點和描述子
keypoints, descriptors = surf.detectAndCompute(img, None)# 繪制關鍵點
img_kp = cv2.drawKeypoints(img, keypoints, None, (0,255,0), flags=cv2.DRAW_MATCHES_FLAGS_DRAW_RICH_KEYPOINTS)cv2.imshow("SURF Features", img_kp)
cv2.waitKey(0)
cv2.destroyAllWindows()

C++ 示例

#include <opencv2/opencv.hpp>
#include <opencv2/xfeatures2d.hpp>int main()
{cv::Mat img = cv::imread("image.jpg", cv::IMREAD_GRAYSCALE);if (img.empty()) {std::cerr << "Cannot load image!" << std::endl;return -1;}// 創建 SURF 檢測器cv::Ptr<cv::xfeatures2d::SURF> surf = cv::xfeatures2d::SURF::create(400); // Hessian thresholdstd::vector<cv::KeyPoint> keypoints;cv::Mat descriptors;surf->detectAndCompute(img, cv::noArray(), keypoints, descriptors);cv::Mat output;cv::drawKeypoints(img, keypoints, output, cv::Scalar::all(-1), cv::DrawMatchesFlags::DRAW_RICH_KEYPOINTS);cv::imshow("SURF", output);cv::waitKey(0);return 0;
}

特征提取算法比較總結

算法是否開源描述子類型匹配類型特點
SIFT?浮點(128維)L2(歐氏距離)穩定性好,適合圖像拼接
SURF?(專利)浮點L2快速但已過時
ORB?二值(256位)Hamming(漢明)快速,適合實時系統
AKAZE?二值或浮點Hamming/L2穩定性與速度兼顧
BRISK?二值Hamming實時性強

小結

  • 推薦優先使用 ORBAKAZE(開源 + 高速);
  • 高精度圖像處理可考慮 SIFT
  • 所有算法都可配合 BFMatcherFLANN 進行匹配;
  • 選擇依據:速度要求匹配精度是否支持縮放旋轉

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/pingmian/89935.shtml
繁體地址,請注明出處:http://hk.pswp.cn/pingmian/89935.shtml
英文地址,請注明出處:http://en.pswp.cn/pingmian/89935.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

復雜度+包裝類型+泛型

什么是集合框架什么是數據結構什么是算法時間復雜度與空間復雜度的概念時間復雜度的表達方式時間復雜度的大 O 的漸近表示法時間復雜度函數的大小排序包裝類和泛型基本數據類型和對應的包裝類型包裝類型出現的原因什么叫做裝箱&#xff08;裝包&#xff09;和拆箱&#xff08;拆…

硬件設計學習DAY15——自舉電容:MOSFET高端驅動的核心奧秘

每日更新教程&#xff0c;評論區答疑解惑&#xff0c;小白也能變大神&#xff01;" 目錄 一.自舉電容 1.自舉電容的作用 2.自舉電路原理 3.工作過程分析 4.實際應用中的問題 5.關鍵要點 二.自舉電容實現MOSFET高端驅動 2.1MOSFET半橋高端驅動的基本原理 2.2自舉電…

【SpringAI實戰】實現仿DeepSeek頁面對話機器人

一、實現效果 二、代碼實現 2.1 后端代碼 2.2 前端代碼 一、實現效果 可以保存聊天記錄與會話記錄 二、代碼實現 2.1 后端代碼 pom.xml <!-- 繼承Spring Boot父POM&#xff0c;提供默認依賴管理 --><parent><groupId>org.springframework.boot</grou…

RedisJSON 指令精講JSON.STRLEN 高效統計字符串長度

1 場景與價值 在日志累加、指標采集、消息追蹤等場景中&#xff0c;我們常需快速判斷某個字符串字段“到底有多長”&#xff0c;以便&#xff1a; 阻止過大日志&#xff1a;若長度超限則截斷或歸檔&#xff1b;動態分桶&#xff1a;按長度選擇不同存儲策略&#xff1b;性能監控…

大數據量查詢計算引發數據庫CPU告警問題復盤

大數據量查詢計算引發數據庫CPU告警問題復盤一、背景二、根因分析三、解決方案方案1&#xff1a;多線程緩存方案2&#xff1a;利用中間表緩存四、總結一、背景 2025年7月份某天&#xff0c;CDP系統每天不定時推送我們的Portal服務&#xff0c;生產環境運營看板會展示統計數據&…

2025最新版虛幻引擎5(UE5)C++入門教程:前言——你的隨身教程和學習筆記

大家好&#xff0c;我是開發游戲的老王&#xff0c;一名高校教師&#xff0c;我主講游戲開發已有十余年時間&#xff0c;通過我的博客大家應該可以了解我所涉獵的游戲技術范疇非常廣泛&#xff0c;除了Unreal,Unity,Godot等主流游戲引擎&#xff0c;還包括Blender、Houdini、3D…

(3)重定向 | 時間相關指令 | 文件查找 | 打包與壓縮

Ⅰ . 初始重定向01 輸出重定向 >在上一節中我們為了方便講解 head 和 tail 指令&#xff0c;我們用到了 > 去生成了一千行文本。通過 > 將生成的一千行文本寫入到了 large.txt 中……我們現在來正式介紹一下&#xff1a;$ echo "內容" > [目標] 本來應…

DTH11測量溫濕度學習(第十一天)

&#x1f468;?&#x1f4bb;個人主頁&#xff1a;開發者-削好皮的Pineapple! &#x1f468;?&#x1f4bb; hello 歡迎 點贊&#x1f44d; 收藏? 留言&#x1f4dd; 加關注?! &#x1f468;?&#x1f4bb; 本文由 削好皮的Pineapple! 原創 &#x1f468;?&#x1f4…

Go語言初識--標識符 可見性

Go語言初識–標識符 可見性 和C語言相似&#xff0c;go語言的基本組成有&#xff1a; 包聲明&#xff0c;編寫源文件時&#xff0c;必須在非注釋的第一行指明這個文件屬于哪個包&#xff0c;如package main。引入包&#xff0c;其實就是告訴Go 編譯器這個程序需要使用的包&…

Python實例之畫小豬佩奇

效果圖&#xff1a;python代碼以及解釋&#xff0c;沒有運用模塊。 """ 繪制小豬佩奇 """ from turtle import *def nose(x,y):"""畫鼻子"""penup()# 將海龜移動到指定的坐標goto(x,y)pendown()# 設置海龜的方向&…

Unity筆記——事件中心

事件中心是什么事件中心是 Unity 游戲開發中常用的架構設計&#xff0c;它基于觀察者模式 或 發布-訂閱模式&#xff0c;通過委托和事件構建的一種消息管理系統。主要用于降低代碼耦合度&#xff0c;實現模塊間的松耦合通信的消息處理系統能大幅提升代碼的可維護性和擴展性&…

Java: 反射機制的 ParameterizedType(參數化類型)

在 Java 中&#xff0c;ParameterizedType 是 java.lang.reflect 包下的一個接口&#xff0c;屬于反射 API 的一部分&#xff0c;主要用于表示參數化類型&#xff08;即帶有類型參數的泛型類型&#xff09;。它是 Java 反射機制中處理泛型類型信息的關鍵接口之一。 一、什么是參…

OkHttp 與 Retrofit 完美結合:打造高效的 Android 網絡請求

前言在現代 Android 開發中&#xff0c;網絡請求是幾乎每個應用都必不可少的功能。OkHttp 和 Retrofit 作為當前最流行的網絡請求庫組合&#xff0c;為開發者提供了簡潔高效的解決方案。本文將詳細介紹如何將這兩者結合使用&#xff0c;充分發揮它們的優勢。一、OkHttp 和 Retr…

系統辨識建模

系統辨識建模 一、系統辨識建模的作用 1. 建立真實物理系統的數學模型 2. 為后續控制器/強化學習算法提供仿真環境 3. 提高控制精度和安全性 二、本文的系統辨識是怎么做的 1. 實驗采集 2. 數學建模 3. 在控制系統中的作用 三、實際用法流程(簡化版) 1. 系統辨識階段 2. 強化…

Android開發:Java與Kotlin深度對比

1. 語言特性與現代性 Java (特別是 Android 主要使用的 Java 8 及之前版本): 相對冗長&#xff1a; 需要編寫更多的樣板代碼&#xff08;如 getter/setter、findViewById 的顯式類型轉換、匿名內部類等&#xff09;。空指針異常 (NPE)&#xff1a; 類型系統默認允許 null&#…

米家打印機驅動:Wi-Fi 無線打印絲滑順暢不卡頓,從此告別對打印機干瞪眼

各位小米家居控們&#xff0c;你們有沒有過這種經歷&#xff0c;新買的打印機回家&#xff0c;結果電腦跟它像倆傲嬌的小情侶&#xff0c;死活不搭話&#xff1f;急得你想當場表演一個“打印機抱頭痛哭”&#xff1f;別急&#xff0c;今天就給你們安利個神隊友——米家打印機驅…

日語學習-日語知識點小記-構建基礎-JLPT-N3階段(7):自動詞 & 他動詞

日語學習-日語知識點小記-構建基礎-JLPT-N3階段&#xff08;7&#xff09;&#xff1a;自動詞 &#xff06; 他動詞1、前言&#xff08;1&#xff09;情況說明&#xff08;2&#xff09;工程師的信仰2、知識點&#xff08;&#xff11;&#xff09;自動詞 &#xff06; 他動…

深入理解設計模式:訪問者模式詳解

在軟件開發中&#xff0c;我們經常會遇到需要對一個復雜對象結構進行操作的情況。隨著需求的不斷變化&#xff0c;我們可能需要在這個對象結構上添加各種新的操作。如果直接在對象結構中添加這些操作&#xff0c;會導致類的職責過重&#xff0c;且每次添加新操作都需要修改原有…

Linux timerfd 定時器封裝

使用 timerfd epoll() 實現&#xff0c;簡潔精確。沒定義 MU_ERROR 宏的話替換為 printf 即可。mu_timer.h:#ifndef _MU_TIMER_H_ #define _MU_TIMER_H_#ifdef __cplusplus extern "C" { #endif#include <stdint.h> #include <time.h> #include <pth…

【樣式效果】Vue3實現仿制iOS按鈕動態效果

iOS開關效果定義變量&#xff1a; <style scoped lang"scss">.layout {// 按鈕寬度$button-width: 500px;// 按鈕高度$button-height: 250px;// 按鈕里面圓形直徑$circle-diameter: 200px;// 按鈕背景與里面圓形間距$button-circle-offset:calc(($button-he…