OpenCV-Python(39):Meanshift和Camshift算法

目标

  • 学习了解Meanshift 和Camshift 算法
  • 在视频中找到并跟踪目标

Meanshift

原理

        Meanshift算法是一种基于密度的聚类算法,用于将数据点划分为不同的类别。它的原理是通过数据点的密度分布来确定聚类中心,然后将数据点移动到离其最近的聚类中心,并不断迭代这个过程,直到收敛为止。假设我们有一堆点(比如直方图反向投影得到的点)和一个小的圆形窗口,我们要完成的任务就是将这个窗口移动到最大灰度密度处(或者是点最多的地方)。如下图所示:

        初始窗口是蓝色的"C1",它的圆心为蓝色方框“C1_o”,而窗口中所有点质心却是“C1_r”(小的蓝色圆圈),很明显圆心和点的质心没有重合。所以移动圆心C1_o 到质心C1_r,这样我们就得到了一个新的窗口。此时又可以找到新窗口内所有点的质心,大多数情况下还是不重合的,所以重复上述的操作:将新窗口的中心移动到新的质量心。就这样不停跌代操作直到窗口的中心和其所包含点的质心重合为止或者有一点小的误差。按照这样的操作我们的窗口最终会落在像素值(和)最大的地方。如上图所示:C2是窗口的最后位址,我们可以看出来这个窗口中的像素点最多。整个过程如下图所示:

        通常情况下我们要使用直方图方向投影得到的图像和目标对象的起始位置。目标对像的移动会反映到直方图反向投影图中。就这样meanshift 算法就把我们的窗口移动到图像中灰度密度最大的区域了。 

步骤

下面是Meanshift算法的步骤:

1.初始化每个数据点的聚类中心为其自身的位置。
2.对于每个数据点,计算它与其他数据点的距离,并根据距离确定一个窗口大小。
3.在窗口内计算数据点的质心(即所有数据点的平均位置),作为新的聚类中心。
4.将数据点移动到离其最近的聚类中心,并更新聚类中心的位置。
5.重复步骤3和步骤4,直到聚类中心的位置不再改变或达到最大迭代次数。

优缺点及结论

        Meanshift算法的优点是不需要预先指定聚类的个数,而是通过数据点的密度分布自动确定聚类中心。它在处理非线性、非凸的数据分布时表现良好,并且对初始聚类中心的选择不敏感。然而,Meanshift算法也有一些限制。首先,它对大规模数据集的处理效率较低。其次,算法的收敛速度较慢,可能需要较多的迭代次数才能收敛。此外,Meanshift算法对窗口大小的选择比较敏感,不同的窗口大小可能导致不同的聚类结果。总的来说,Meanshift算法是一种简单而有效的聚类算法,适用于处理中等规模的数据集和非线性、非凸的数据分布。它在计算机视觉、图像分割等领域有广泛的应用。

OpenCV中的Meanshift

        在OpenCV 中使用Meanshift 算法首先我们要对目标对象进行设置,计算目标对象的直方图。这样在执行 meanshift 算法时我们就可以将目标对象反向投影到每一帧中去了。另外我们还需要提供窗口的起始位置。在这里我们计算H(Hue)通道的直方图,同样为了避免低亮度造成的影响,我们使用函数cv2.inRange() 将低亮度的值忽略掉。

import numpy as np
import cv2
cap = cv2.VideoCapture('slow.flv')
# take first frame of the video
ret,frame = cap.read()
# setup initial location of window
r,h,c,w = 250,90,400,125 # simply hardcoded the values
track_window = (c,r,w,h)
# set up the ROI for tracking
roi = frame[r:r+h, c:c+w]
hsv_roi = cv2.cvtColor(frame, cv2.COLOR_BGR2HSV)
mask = cv2.inRange(hsv_roi, np.array((0., 60.,32.)), np.array((180.,255.,255.)))
roi_hist = cv2.calcHist([hsv_roi],[0],mask,[180],[0,180])
cv2.normalize(roi_hist,roi_hist,0,255,cv2.NORM_MINMAX)
# Setup the termination criteria, either 10 iteration or move by atleast 1 pt
term_crit = ( cv2.TERM_CRITERIA_EPS | cv2.TERM_CRITERIA_COUNT, 10, 1 )
while(1):
    ret ,frame = cap.read()
    if ret == True:
        hsv = cv2.cvtColor(frame, cv2.COLOR_BGR2HSV)
        dst = cv2.calcBackProject([hsv],[0],roi_hist,[0,180],1)
        # apply meanshift to get the new location
        ret, track_window = cv2.meanShift(dst, track_window, term_crit)
        # Draw it on image
        x,y,w,h = track_window
        img2 = cv2.rectangle(frame, (x,y), (x+w,y+h), 255,2)
        cv2.imshow('img2',img2)
        k = cv2.waitKey(60) & 0xff
        if k == 27:
            break
        else:
            cv2.imwrite(chr(k)+".jpg",img2)
    else:
        break
cv2.destroyAllWindows()
cap.release()

下面是我使用meanshift 算法对一个视频前三帧分析的结果:

 Camshift

原理

        你认真看上面的结果了吗?这里面还有一个问题。我们的窗口的大小是固定的,而汽车由远及近在(视觉上)是一个逐渐变大的过程,固定的窗口是不合适的。所以我们需要根据目标的大小和角度来对窗口的大小和角度进行修订。OpenCVLabs 为我们带来的(解决方案)1988 年提出一个叫做CAMshift 的算法。

        Camshift算法是一种基于Meanshift算法的目标跟踪算法,用于实时跟踪视频中的运动目标。它在Meanshift算法的基础上做了一些改进,可以自适应地调整窗口大小,并且可以处理目标的尺度变化和旋转。这个算法首先使用meanshift算法找到(并覆盖)目标之后,再去调整窗口的大小,

它还会计算目标对象的最佳外接椭圆的角度并以此调节窗口角度。然后使用更新后的窗口大小和角度来在原来的位置继续meanshift。重复这个过程直到达到徐亚的精度。

步骤 

Camshift算法的步骤如下:

1.初始化目标区域。首先,选择一个初始的目标区域作为跟踪目标,可以是手动选择或者通过其他目标检测算法获得。
2.计算目标的颜色直方图。在目标区域内,计算目标的颜色直方图作为目标的特征表示。
3.在每一帧中,计算目标区域的颜色直方图。根据上一帧的目标位置和窗口大小,计算当前帧中目标区域的颜色直方图。
4.计算当前帧中目标区域的质心。根据窗口大小和颜色直方图,使用Meanshift算法计算当前帧中目标区域的质心,并将其作为新的目标位置。
5.更新窗口大小。根据目标区域的质心和颜色直方图,计算新的窗口大小,并进行缩放和旋转。
6.重复步骤3到步骤5,直到目标区域的位置不再改变或达到最大迭代次数。

优缺点及结论

       Camshift算法的核心是通过颜色直方图计算目标区域的质心,并通过Meanshift算法不断迭代来跟踪目标。它优点是可以自适应地调整窗口大小,适应目标的尺度变化和旋转。此外,Camshift算法对初始目标区域选择相对不敏感,可在一定程度上处理目标遮挡和背景干扰。然而,Camshift算法也有一些限制。首先,它对目标的颜色敏感,当目标的颜色变化较大时可能会导致跟踪失败。其次,算法对目标的形状变化不敏感,难以跟踪形状变化较大的目标。总的来说,Camshift算法是一种简单而有效的目标跟踪算法,适用于处理实时视频中的目标跟踪任务。它在计算机视觉、机器人导航、交通监控等领域有广泛的应用。

OpenCV中的Camshift

        与Meanshift 基本一样,但是返回的结果是一个带旋转角度的矩形(这是我们的结果)以及这个矩形的参数(被用到下一次迭代过程中)。下面是代码:

import numpy as np
import cv2

cap = cv2.VideoCapture('slow.flv')

# take first frame of the video
ret,frame = cap.read()

# setup initial location of window
r,h,c,w = 250,90,400,125 # simply hardcoded the values
track_window = (c,r,w,h)

# set up the ROI for tracking
roi = frame[r:r+h, c:c+w]
hsv_roi = cv2.cvtColor(frame, cv2.COLOR_BGR2HSV)
mask = cv2.inRange(hsv_roi, np.array((0., 60.,32.)), np.array((180.,255.,255.)))
roi_hist = cv2.calcHist([hsv_roi],[0],mask,[180],[0,180])
cv2.normalize(roi_hist,roi_hist,0,255,cv2.NORM_MINMAX)

# Setup the termination criteria, either 10 iteration or move by atleast 1 pt
term_crit = ( cv2.TERM_CRITERIA_EPS | cv2.TERM_CRITERIA_COUNT, 10, 1 )

while(1):
    ret ,frame = cap.read()
    if ret == True:
        hsv = cv2.cvtColor(frame, cv2.COLOR_BGR2HSV)
        dst = cv2.calcBackProject([hsv],[0],roi_hist,[0,180],1)
        # apply meanshift to get the new location
        ret, track_window = cv2.CamShift(dst, track_window, term_crit)
        # Draw it on image
        pts = cv2.boxPoints(ret)
        pts = np.int0(pts)
        img2 = cv2.polylines(frame,[pts],True, 255,2)
        cv2.imshow('img2',img2)
        k = cv2.waitKey(60) & 0xff
        if k == 27:
            break
        else:
            cv2.imwrite(chr(k)+".jpg",img2)
    else:
        break
cv2.destroyAllWindows()
cap.release()

 对三帧图像分析的结果如下:

camshift的交互式演示

        OpenCV官方提供了一个交互式演示代码来展示Camshift算法的效果。以下是示例代码的主要步骤:

1.导入必要的库和模块:

import numpy as np
import cv2

2.定义鼠标事件的回调函数:

def mouse_callback(event, x, y, flags, param):
    # 处理鼠标事件

3.创建窗口和摄像头对象:

cv2.namedWindow("CamShift Demo")
capture = cv2.VideoCapture(0)

4.初始化一些变量:

# 设置初始目标区域
selection = None
drag_start = None
tracking_state = 0

5.定义鼠标事件回调函数的具体实现:

def mouse_callback(event, x, y, flags, param):
    global selection, drag_start, tracking_state

    if event == cv2.EVENT_LBUTTONDOWN:
        drag_start = (x, y)
        tracking_state = 0

    if drag_start:
        if flags & cv2.EVENT_FLAG_LBUTTON:
            h, w = frame.shape[:2]
            xo, yo = drag_start
            x0, y0 = np.minimum(xo, x), np.minimum(yo, y)
            x1, y1 = np.maximum(xo, x), np.maximum(yo, y)
            x0, y0 = np.maximum(0, x0), np.maximum(0, y0)
            x1, y1 = np.minimum(w, x1), np.minimum(h, y1)
            selection = None
            if x1-x0 > 0 and y1-y0 > 0:
                selection = (x0, y0, x1, y1)
        else:
            drag_start = None
            if selection is not None:
                tracking_state = 1

6.在循环中处理每一帧图像并进行Camshift跟踪:

while True:
    # 读取一帧图像
    ret, frame = capture.read()
    if not ret:
        break

    # 复制一份原始图像
    vis = frame.copy()

    # 转换为HSV颜色空间
    hsv = cv2.cvtColor(frame, cv2.COLOR_BGR2HSV)

    # 如果已经选择了目标区域,则进行Camshift跟踪
    if selection is not None:
        # 提取目标区域的颜色直方图
        x0, y0, x1, y1 = selection
        hsv_roi = hsv[y0:y1, x0:x1]
        mask = cv2.inRange(hsv_roi, np.array((0., 60., 32.)), np.array((180., 255., 255.)))
        roi_hist = cv2.calcHist([hsv_roi], [0], mask, [16], [0, 180])
        cv2.normalize(roi_hist, roi_hist, 0, 255, cv2.NORM_MINMAX)

        # 使用Camshift算法进行跟踪
        term_crit = (cv2.TERM_CRITERIA_EPS | cv2.TERM_CRITERIA_COUNT, 10, 1)
        hsv_backproj = cv2.calcBackProject([hsv], [0], roi_hist, [0, 180], 1)
        ret, track_window = cv2.CamShift(hsv_backproj, (x0, y0, x1-x0, y1-y0), term_crit)

        # 绘制跟踪结果
        pts = cv2.boxPoints(ret)
        pts = np.int0(pts)
        cv2.polylines(vis, [pts], True, (0, 255, 0), 2)

    # 如果正在拖动选择目标区域,则绘制选择框
    if drag_start:
        x, y = drag_start
        cv2.rectangle(vis, (x, y), (x1, y1), (0, 255, 0), 2)
        cv2.imshow("CamShift Demo", vis)

    # 显示图像并等待按键
    cv2.imshow("CamShift Demo", vis)
    if cv2.waitKey(1) == 27:
        break

        这个示例代码展示了如何使用Camshift算法进行目标跟踪,并通过鼠标事件实现了交互式的目标选择和跟踪。你可以在这个基础上进行修改和扩展,以适应你的具体应用场景。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/321983.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

UIAlertController简单使用-swift

UIAlertControlle时IOS的对话框控制器(警报控制器),简单使用方法如下: 步骤都一样,先是创建UIAlertController,然后创建UIAlertAction,再将UIAlertAction添加到UIAlertController中,…

PyTorch深度学习实战(30)——Deepfakes

PyTorch深度学习实战(30)——Deepfakes 0. 前言1. Deepfakes 原理2. 数据集分析3. 使用 PyTorch 实现 Deepfakes3.1 random_warp.py3.2 Deepfakes.py 小结系列链接 0. 前言 Deepfakes 是一种利用深度学习技术生成伪造视频和图像的技术。它通过将一个人的…

TongLINKQ(2):TongLINKQ服务端安装

1 安装前的准备 明确应用(JDK)和TongLINK/Q的版本、位数(要么都是32位,要么都是64位)TLQ安装程序使用InstallAnywhere打包而成,因此需要JDK1.5及以上版本。 2 安装步骤 3 选择安装安装包 目前TongLINKQ的…

相信我,努力真的有用!

2023年对很多人来说都是异常艰辛的一年,大环境下的每个人都面对着或多或少的挑战,在这一年的时间里,身边的朋友陆陆续续的跳槽、创业、再就业,结婚,生娃,回老家,每个人渐渐的在时代面前或妥协或…

快速入门Java NIO(New I/O)的网络通信框架--Netty

Netty 入门 了解netty前需要对nio有一定认识,该笔记基础来自bilinbili黑马,在此基础上自己学习的笔记,添加了一些自己的理解 了解java 非阻塞io编程 1. 概述 1.1 Netty 是什么? Netty is an asynchronous event-driven network application framework for rapid …

Linux系统的通配符* ,你可能还不知道的一些规则 。

最近老被同学问到关于通配符操作的问题,本来觉得是一个很简单的问题,结果它和其它命令一结合,就很难给出合理的理解 。进行了很多实验,总是难以有个满意的答案 。于是决定要好好研究一番 ,终于在多次的测试和验证过程中…

ElasticSearch分布式搜索引擎(两万字详解)

文章目录 ElasticSearch分布式搜索引擎1.了解ESELK技术栈elasticsearch和lucene为什么不是其他搜索技术?总结倒排索引正向索引倒排索引正向和倒排 es的一些概念文档和字段索引和映射mysql与elasticsearch 2.安装elasticsearch1.部署单点es1.1.创建网络1.2.拉取镜像1…

N5181A/安捷伦Agilent N5181A信号发生器

181/2461/8938产品概述: 规格(说明书):表示已校准的仪器在工作温度范围0-55C内存放至少2小时,除非另有说明,并经过45分钟预热期后的保证性能。的指标包括测量不确定度。除非另有说明,本文档中的…

2023年度总结 - 职业生涯第一个十年

2023年只剩下最后一周,又到了一年一度该做年末总结的时候了。 回想起去年,还有人专门建立了一个关于年度总结文章汇总的仓库。读了很多篇别人写的,给了我很多的触动和感想。这里的每篇文章都是关于某个人这一整年的生活和工作的轨迹啊。即使你…

12-桥接模式(Bridge)

意图 将抽象部分与它的实现部分分离,使他们可以独立地变化 个人理解 一句话概括就是只要是在抽象类中聚合了某个接口或者抽象类,就是使用了桥接模式。 抽象类A中聚合了抽象类B(或者接口B),A的子类的方法中在相同的场…

流程图用什么软件做?五款优质在线绘制工具看一看

流程图用什么软件做?现在,流程图已经成为了我们工作中不可或缺的工具。它能够清晰地展示各个步骤之间的关系,使我们更好地理解并优化流程。那么,在众多的流程图绘制工具中,哪一款最适合你呢?下面就给大家介…

【办公技巧】ppt修改全部字体怎么改?

制作完PPT之后,想要更换ppt中的字体,有没有什么快捷的方法呢?今天分享两个方法,一键修改ppt文件字体。 方法一: 找到功能栏中的编辑选项卡,点击替换 – 替换字体,在里面选择我们想要替换的字体…

单细胞转录组学对代谢功能障碍相关脂肪变性肝病的类器官模型进行分析

前言 最近接触比较多肝纤维化项目,包括空转、单细胞和普通的BULK转录组,本文是肝脏疾病类器官构建,所以结果是比较确定的,只是对比不同处理和培养哪种效果更好,适合了解纤维化进展和哪些分子和细胞参与,以…

数据可视化大屏自适应,保持比例不变形,满足不同分辨率的需求——利用transform的scale属性缩放,缩放整个页面。

文章目录 一、需求背景:二、需求分析:三、选择方案:四、实现代码:五、效果预览:六、封装组件: 一、需求背景: 数据可视化大屏是一种将数据、信息和可视化效果集中展示在一块或多块大屏幕上的技…

项目中使用iframe引入html 解决路由错乱问题以及父子组件传值调用方法

iframe与外部之间传值 父组件 <iframeid"iframe"src"luckysheet/index.html"frameborder"0"scrolling"no"style"width: 100%; height: 60vh; border: 0"/>const frame document.getElementById(iframe);frame.onloa…

8. 《自动驾驶与机器人中的SLAM技术》基于保存的自定义NDT地图文件进行自动驾驶车辆的激光定位

目录 1. 为 NDT 设计一个匹配度评估指标&#xff0c;利用该指标可以判断 NDT 匹配的好坏。 2. 利用第 1 题的指标&#xff0c;修改程序&#xff0c;实现 mapping 部分的回环检测。 3. 将建图结果导出为 NDT map&#xff0c;即将 NDT 体素内的均值和协方差都存储成文件。 4.…

学习记录11-SPI通信(软件)

目录 前言 一、引脚定义 二、代码 1.初始化 2.操作代码 三、验证 前言 对SPI进行结构封装&#xff0c;方便使用。方便讲解&#xff0c;用W&#xff12;&#xff15;Q&#xff16;&#xff14;芯片进行讲解 一、引脚定义 #define SPI_CS_PROT GPIOB //CS接线引脚通道…

CAN总线记录仪在车企服务站的应用

CAN总线记录仪在车企服务站的应用 CAN总线记录仪在车企服务站中有着广泛的应用。这种设备可以记录车上的CAN总线数据&#xff0c;方便工程师进行分析&#xff0c;以找出可能存在的问题。CAN记录仪一般采用TF卡来存储数据&#xff0c;实现离线脱机实时存储。数据存储完毕后&…

【python】12.字符串和正则表达式

使用正则表达式 正则表达式相关知识 在编写处理字符串的程序或网页时&#xff0c;经常会有查找符合某些复杂规则的字符串的需要&#xff0c;正则表达式就是用于描述这些规则的工具&#xff0c;换句话说正则表达式是一种工具&#xff0c;它定义了字符串的匹配模式&#xff08;…
最新文章