OpenCV 背景差分法

OpenCV 背景差分法 Background Subtraction Methods(BS)

在这篇博文中,您将学会如何用Python调用OpenCV,进行背景差分(Background Subtraction)。

在这里插入图片描述

背景差分法是一种被广泛应用的技术,我们可以利用背景差分法,通过架设的固定位置的相机(static camera)来得到前景或者移动目标在图像中所在的位置(foreground mask)。

就像背景差分法这个名字所说的那样,这种方法是通过把当前帧(current frame)的每一个像素与背景模板(background model)每一个像素做减法,来判断这个像素是属于前景还是背景。

在这里插入图片描述

背景差分法可以应用与很多领域,比如日常的安防监控,我们可以通过背景差分法来判断监控视频中是否有人来回走动,快速的定位有人活动的视频片段。也可以在野外架设摄像头,用这种方法快速定位有野生动物的出没的视频片段!

而且,背景差分法是一种计算资源需求比较小的算法,我们可以在嵌入式设备,比如树莓派(Raspberry Pi)或者Jetson Nano中部署它。

OpenCV中为我们提供了很多种背景差分的方法,我们在实际项目中可以选择一种方法来进行背景差分,方法种类如下:

在这里插入图片描述

从图中我们可以看到,OpenCV中的背景差分法一共分为三类cv下面一类,cv::bgsegm下面一类,还有提供了cuda加速的cv::cuda一类,在这篇博文中,我将向大家介绍最常用的MOG2和KNN这两种方法。

BackgroundSubtractorMOG2

这个是以高斯混合模型(GMM)为基础的背景/前景分割算法(关于GMM,大家可以看两篇知乎的帖子:一文详解高斯混合模型原理高斯混合模型(GMM))。它是以 2004 年和 2006 年 Z.Zivkovic 的两篇文章为基础的,分别是"Improved adaptive Gaussian mixture model for background subtraction" 和 “Efficient Adaptive Density Estimation per Image Pixel for the Task of Background Subtraction” 。这个算法的一个特点是它为每一个像素选择一个合适数目的高斯分布。。它能更好地适应光照不同等各种场景。 我们需要创建一个背景对象。但在这里我们我们可以选择是否检测阴影。如果 detectShadows = True(默认值),它就会检测并将影子标记出来,但是这样做会降低处理速度。影子会被标记为灰色。

import numpy as np
import cv2 as cv

cap = cv.VideoCapture('vtest.avi')	# read from a video file
# cap = cv.VideoCapture(0)   # read from a webcam
fgbg = cv.createBackgroundSubtractorMOG2()

while(1):
    ret, frame = cap.read()
    fgmask = fgbg.apply(frame)
    cv.imshow('frame',fgmask)
    k = cv.waitKey(30) & 0xff
    if k == 27:
        break

cap.release()
cv.destroyAllWindows()

BackgroundSubtractorKNN

这是一个基于K最近邻(分类算法之邻近算法:KNN(理论篇))的背景前景区分算法,当前景像素数量比较少的时候,算法运行得会非常快。在OpenCV中的调用方式与MOG2很类似,我们只需要把

fgbg = cv.createBackgroundSubtractorMOG2() 换成 fgbg = cv.createBackgroundSubtractorKNN()

即可。或者你如果熟悉argparse,你也可以使用输入变量给python文件的方法来改变使用的背景提取算法。 capture = cv.VideoCapture(0)的时候,就会打开摄像头。

from __future__ import print_function
import cv2 as cv
import argparse

parser = argparse.ArgumentParser(description='This program shows how to use background subtraction methods provided by \
                                              OpenCV. You can process both videos and images.')
parser.add_argument('--input', type=str, help='Path to a video or a sequence of image.', default='vtest.avi')
parser.add_argument('--algo', type=str, help='Background subtraction method (KNN, MOG2).', default='MOG2')
args = parser.parse_args()
if args.algo == 'MOG2':
    backSub = cv.createBackgroundSubtractorMOG2()
else:
    backSub = cv.createBackgroundSubtractorKNN()
capture = cv.VideoCapture(cv.samples.findFileOrKeep(args.input))

if not capture.isOpened:
    print('Unable to open: ' + args.input)
    exit(0)
while True:
    ret, frame = capture.read()

    if frame is None:
        break

    # fgMask = backSub.apply(frame,learningRate=0)
    fgMask = backSub.apply(frame)
    cv.rectangle(frame, (10, 2), (100, 20), (255, 255, 255), -1)
    cv.putText(frame, str(capture.get(cv.CAP_PROP_POS_FRAMES)), (15, 15),
               cv.FONT_HERSHEY_SIMPLEX, 0.5, (0, 0, 0))

    cv.imshow('Frame', frame)
    cv.imshow('FG Mask', fgMask)

    keyboard = cv.waitKey(30)
    if keyboard == 'q' or keyboard == 27:
        break


在这里插入图片描述

fgMask = backSub.apply(frame) 这个函数中learningRate 默认是-1,就是自动调节,这样我们的背景模板是动态变化的,因此,这时候的背景提取算法只对移动的目标感兴趣。

如果我们想要的不是对移动的目标感兴趣,而是对与背景不同的目标感兴趣,我们可以这样做fgMask = backSub.apply(frame, learningRate=0) 这样就会停止背景模板的更新。

在这里插入图片描述

当然,在想这么做之前,要先把摄像头的自动亮度和自动调焦距的功能关掉(可以参考我的另一篇CSDN),否则噪声会非常多。更多的关于OpenCV 背景差分法和更多OpenCV 的知识,欢迎大家到我的视频课《OpenCV从萌新到专家》中收看,谢谢大家的支持!

  • 23
    点赞
  • 96
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
### 回答1: 背景差分法是一种常用的计算机视觉技术,用于检测视频中的运动物体。在OpenCV-Python中,可以使用cv2.BackgroundSubtractor类来实现背景差分法。该类提供了几种不同的背景建模算法,包括基于高斯混合模型的算法和基于自适应混合高斯模型的算法。使用背景差分法可以实现实时的运动物体检测,例如在视频监控系统中检测入侵者。 ### 回答2: 背景差分法(Background Subtraction)是一种常见的计算机视觉技术,用于提取影像序列中的前景目标。该技术通过对比当前图像和背景图像,找出两者之间的差异来识别前景目标。随着计算机视觉的发展,背景差分法的应用越来越广泛。 在OpenCV-Python中,可以使用BackgroundSubtractor类来实现背景差分处理。该类通过调用createBackgroundSubtractorMOG2方法来实例化一个背景分割器,即BackgroundSubtractorMOG2类的对象。其核心算法使用了高斯混合模型(Gaussion Mixture Model)进行背景建模,从而识别前景物体。 在实际使用过程中,首先需要对输入视频流或图片序列进行预处理,消除一些噪点或干扰信息。然后,通过调用BackgroundSubtractor类处理每一帧图像,得到前景图像。最后,可以进行二值化或形态学处理来生成最终的前景目标图像。 需要注意的是,背景差分法在处理过程中可能会出现一些问题,例如光照变化、背景变化、阴影等。因此,在实际应用中需要选择适当的参数和处理策略,以达到更好的结果。 总之,通过OpenCV-Python中的背景差分法技术,可以很好地实现影像序列的前景目标提取,为计算机视觉的研究和应用提供了便利。 ### 回答3: 背景差分法是一种常用的目标检测算法,其原理是将动态目标和静态背景进行分离。在opencv-python中,背景差分法的实现方式有多种,其中最常用的是利用BackgroundSubtractor类进行背景建模,并通过其apply方法进行背景差分计算。 首先,通过BackgroundSubtractor类对背景进行建模,以得到背景图像。通常情况下,背景图像是通过一定时间段内采集的图像帧平均或者高斯混合等方式得到的。背景图像在建模完成后一般不再重新计算,而是随着时间的推移而更新,以保持背景随时间变化的一致性。 其次,通过apply方法计算背景差分,并进行二值化处理,以得到二值化的前景掩码图像。二值化处理通常采用阈值法或自适应阈值法等常用的二值化方法。 最后,对前景掩码图像进行形态学操作,以去除噪声和小区域的干扰,同时对前景目标进行连通域分析,以得到各个前景目标的位置、形状等信息。形态学操作常用的有膨胀、腐蚀等方法;连通域分析常用的有轮廓检测、面积统计等方法。 需要注意的是,背景差分法虽然简单有效,但在实际应用中也存在一些限制和应对措施。例如,由于前景区域的形状、位置、大小等变化,以及背景的光照等因素,背景建模和前景掩码图像的计算都可能产生较大误差。为了降低误差,可以采用多种后处理方法,如帧间差分、自适应背景更新、各种形态学操作等。同时,还可以通过多帧融合、多种算法融合等方式进一步提高检测的准确性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值