一直想基于传统图像匹配方式做一个融合Demo,也算是对上个阶段学习的一个总结。

由此,便采购了一个摄像头,在此基础上做了实时检测平面目标的特征匹配算法。

代码如下:

# coding: utf-8
'''
@author: linxu
@contact: 17746071609@163.com
@time: 2021-07-26 上午11:54
@desc: 基于特征匹配的实时平面目标检测算法
@Ref: https://docs.opencv.org/3.0-beta/doc/py_tutorials/py_feature2d/py_feature_homography/py_feature_homography.htm
''' import numpy as np
import cv2 class ObjectDetector:
""" 基于特征匹配的实时平面目标检测算法 """
def __init__(self):
# 特征点检测-选择不同的特征描述子
self.feature_detector = cv2.AKAZE_create() # Use AKAZE
# self.feature_detector = cv2.ORB_create() # Use ORB
# self.feature_detector = cv2.KAZE_create()# Use KAZE
# self.feature_detector = cv2.SIFT_create()# Use SIFT
# self.feature_detector = cv2.BRISK_create()# Use BRISK # 摄像头相机参数设置VideoCapture
self.vidcap = cv2.VideoCapture(0)
self.vidcap.set(3, 640) # 宽度
self.vidcap.set(4, 480) # 高度
self.vidcap.set(5, 15) # 帧率 # 通过ROI(感兴趣区域)来注册目标对象
self.sub_topleft = [100, 220] # [0, 0] # [y,x]100 220
self.sub_width = 200 #640 200
self.sub_height = 200 #480 200
self.sub_bottomright = [self.sub_topleft[0] + self.sub_height - 1,\
self.sub_topleft[1] + self.sub_width - 1]
# rect矩形框体
self.rect_color = (0, 255, 0) # green
self.rect_thickness = 3
self.rect_tl_outer_xy = (self.sub_topleft[1] - self.rect_thickness, self.sub_topleft[0] - self.rect_thickness)
self.rect_br_outer_xy = (self.sub_bottomright[1] + self.rect_thickness, self.sub_bottomright[0] + self.rect_thickness) # 特征(描述符)向量距离的阈值
self.ratio = 0.75
self.registered = False
self.min_match_count = 5
self.show_rectangle = True def register(self):
"""注册目标对象"""
print("\n将目标物体靠近相机.")
print("确保对象完全覆盖矩形内部(背景不可见).")
print("然后,按“r”注册对象.\n") while self.vidcap.isOpened():
ret, frame = self.vidcap.read()
cv2.rectangle(frame, self.rect_tl_outer_xy, self.rect_br_outer_xy,\
self.rect_color, self.rect_thickness)
cv2.imshow("Registration (press 'r' to register)", frame)
if cv2.waitKey(1) & 0xFF == ord('r'):
# 图像切片
subimg = frame[self.sub_topleft[0]:(self.sub_topleft[0] + self.sub_height),
self.sub_topleft[1]:(self.sub_topleft[1] + self.sub_width)]
self.kp0, self.des0 = self.feature_detector.detectAndCompute(subimg, None)
self.queryimg = subimg
self.registered = True
break def detect(self):
""" 使用特征点查找对象 """
global mask
if not self.registered:
print("Call 'register()' first.")
return print("Start detection...")
print("按“q”退出.")
print("按“h”隐藏绿色矩形.\n") # 声明一个暴力匹配器Blute-Force (BF) matcher
bf = cv2.BFMatcher() while self.vidcap.isOpened():
ret, frame = self.vidcap.read() # 关键点(kp)检测和计算描述符(des)
kp, des = self.feature_detector.detectAndCompute(frame, None) # 在关键点之间应用knn匹配
matches = bf.knnMatch(self.des0, des, k=2) # 根据阈值筛选关键特征点
# good = [[m] for m, n in matches if m.distance < self.ratio * n.distance]
good = []
for m, n in matches:
if m.distance < self.ratio * n.distance:
good.append([m])
print('len',len(good)) contours = []
# 查找单应性矩阵
if (len(good) > self.min_match_count) and self.show_rectangle:
# 建立坐标矩阵
src_pts = np.float32([self.kp0[m[0].queryIdx].pt for m in good]).reshape(-1, 1, 2)
dst_pts = np.float32([kp[m[0].trainIdx].pt for m in good]).reshape(-1, 1, 2) # 计算多个二维点对之间的最优单映射变换矩阵 H
M, mask = cv2.findHomography(src_pts, dst_pts, cv2.RANSAC, 5.0)
# Assume color camera
# cv2.imshow('queryimg',self.queryimg)
h, w, c = self.queryimg.shape
pts = np.float32([[0, 0], [0, h-1], [w-1, h-1], [w-1, 0]]).reshape(-1, 1, 2)
dst = cv2.perspectiveTransform(pts, M)
# cv2.circle(frame,tuple(dst_pts[0][0]),5, (255,0,0))
# cv2.circle(frame, tuple(dst_pts[1][0]), 5, (0, 255, 0))
# cv2.circle(frame, tuple(dst_pts[2][0]), 5, (0, 0, 255))
# cv2.circle(frame, tuple(dst_pts[3][0]), 5, (255, 255, 0))
# cv2.imshow('circle', frame)
frame = cv2.polylines(frame, [np.int32(dst)], True, (0, 255, 0), 2, cv2.LINE_AA) # 可视化匹配过程
# 绘画参数
# draw_params = dict(flags=2)
draw_params = dict(matchColor=(0, 255, 0), singlePointColor=(0, 0, 255),flags=0)
img = cv2.drawMatchesKnn(self.queryimg, self.kp0, frame, kp, good, None, **draw_params)
cv2.imshow("Detection (press 'q' to quit)", img) key_pressed = cv2.waitKey(1)
if key_pressed & 0xFF == ord('q'):
break if key_pressed & 0xFF == ord('h'):
self.show_rectangle = False def close(self):
""" 释放VideoCapture并销毁windows """
self.vidcap.release()
cv2.destroyAllWindows() if __name__ == '__main__':
obj_detector = ObjectDetector()
obj_detector.register()
obj_detector.detect()
obj_detector.close()

测试效果,如下:


[OpenCV]基于特征匹配的实时平面目标检测算法的更多相关文章

  1. 基于候选区域的深度学习目标检测算法R-CNN,Fast R-CNN,Faster R-CNN

    参考文献 [1]Rich feature hierarchies for accurate object detection and semantic segmentation [2]Fast R-C ...

  2. 使用Harr特征的级联分类器实现目标检测

    前言  最近在学习人脸的目标检测任务时,用了Haar人脸检测算法,这个算法实现起来太简洁了,读入个.xml,调用函数就能用.但是深入了解我发现这个算法原理很复杂,也很优秀.究其根源,于是我找了好些篇相 ...

  3. 转载:点云上实时三维目标检测的欧拉区域方案 ----Complex-YOLO

    感觉是机器翻译,好多地方不通顺,凑合看看 原文名称:Complex-YOLO: An Euler-Region-Proposal for  Real-time 3D Object Detection ...

  4. 基于模糊Choquet积分的目标检测算法

    本文根据论文:Fuzzy Integral for Moving Object Detection-FUZZ-IEEE_2008的内容及自己的理解而成,如果想了解更多细节,请参考原文.在背景建模中,我 ...

  5. 基于深度学习的目标检测算法:SSD——常见的目标检测算法

    from:https://blog.csdn.net/u013989576/article/details/73439202 问题引入: 目前,常见的目标检测算法,如Faster R-CNN,存在着速 ...

  6. CVPR2020|3D-VID:基于LiDar Video信息的3D目标检测框架

    作者:蒋天园 Date:2020-04-18 来源:3D-VID:基于LiDar Video信息的3D目标检测框架|CVPR2020 Brief paper地址:https://arxiv.org/p ...

  7. 29 基于PCL的点云平面分割拟合算法技术路线(针对有噪声的点云数据)

    0 引言 最近项目中用到了基于PCL开发的基于平面的点云和CAD模型的配准算法,点云平面提取采用的算法如下. 1 基于PCL的点云平面分割拟合算法 2 参数及其意义介绍 (1)点云下采样 1. 参数: ...

  8. 基于COCO数据集验证的目标检测算法天梯排行榜

    基于COCO数据集验证的目标检测算法天梯排行榜 AP50 Rank Model box AP AP50 Paper Code Result Year Tags 1 SwinV2-G (HTC++) 6 ...

  9. 目标检测算法的总结(R-CNN、Fast R-CNN、Faster R-CNN、YOLO、SSD、FNP、ALEXnet、RetianNet、VGG Net-16)

    目标检测解决的是计算机视觉任务的基本问题:即What objects are where?图像中有什么目标,在哪里?这意味着,我们不仅要用算法判断图片中是不是要检测的目标, 还要在图片中标记出它的位置 ...

随机推荐

  1. Learning Spark中文版--第三章--RDD编程(2)

    Common Transformations and Actions   本章中,我们浏览了Spark中大多数常见的transformation(转换)和action(开工).在包含特定数据类型的RD ...

  2. [学习总结]9、Android-Universal-Image-Loader 图片异步加载类库的使用(超详细配置)

    这个图片异步加载并缓存的类已经被很多开发者所使用,是最常用的几个开源库之一,主流的应用,随便反编译几个火的项目,都可以见到它的身影. 可是有的人并不知道如何去使用这库如何进行配置,网上查到的信息对于刚 ...

  3. zabbix之修改中文

    #在zabbix服务器安装中文名包 root@ubuntu:~# sudo apt-get install language-pack-zh* #:修改环境变量 root@ubuntu:~# sudo ...

  4. nodejs代码初探之nodejs启动

    nodejs启动 入口在node_main.cc,解析参数后进入node.cc 中的node::Start() V8::Initialize() //初始化v8SetupProcessObject() ...

  5. POST/GET请求中RequestBody和RequestParam的应用场景

    POST请求时 @RequestBody --> JSON字符串部分 @RequestParam --> 请求参数部分 application/json格局图   图一.png form- ...

  6. 阿里云esc 安装 docker

    1. 更新 yum 到最新: yum update (用 root 用户登录,无需加 sudo,如果不是,需要加,即  yum update ) 2. 安装软件包:yum-util(提供 yum-co ...

  7. String直接赋值与使用new String的区别

    在研究String直接赋值与new String的区别之前我们需要先了解java中的字符串常量池的概念 字符串常量池 String类是我们平常项目中使用频率非常高的一种对象类型,jvm为了提升性能和减 ...

  8. 12.Vue.js 表单

    这节我们为大家介绍 Vue.js 表单上的应用. 你可以用 v-model 指令在表单控件元素上创建双向数据绑定. <div id="app"> <p>in ...

  9. Go modules基础精进,六大核心概念全解析(上)

    点击一键订阅<云荐大咖>专栏,获取官方推荐精品内容,学技术不迷路! Go 语言做开发时,路径是如何定义的?Go Mudules又为此带来了哪些改变?本文将会全面介绍Go modules六大 ...

  10. 手写IOC实践

    一.IOC 1.什么是IOC? 控制反转(英语:Inversion of Control,缩写为IoC),是[面向对象编程]中的一种设计原则,可以用来减低计算机代码之间的[耦合度]其中最常见的方式叫做 ...