概览
最近在做一个项目的时候需要检测摄像头拍摄范围内的图像是否有变化,即需要进行运动检测,在检测到运动之后录制 5 秒的视频存档。
在查阅了一些资料之后,发现了一种效果不错的方法(原始文章见参考内容 1),该方法只进行运动检测,不进行追踪。该方法的主要思路就是计算连续帧之间的差距,如果该差距较大,我们就认为画面的内容发生了变化。其中为了避免假阳性,该方法计算了帧差的标准差。
算法的主要流程如下:
- 计算帧间的距离(Pythagorean distance)
- 在计算出来的距离矩阵上应用高斯模糊
- 使用一定的阈值进行过滤
- 计算标准差
- 标准差大于一定的阈值,认为检测到了运动,输出信息
本文将给出 Python
和 C++
两种语言的实现。
运行环境
操作系统:ARMBIAN 5.38 stable Ubuntu 16.04.3 LTS 3.4.113-sun8i
编程环境:Python 2.7.12
, OpenCV 3.3.0
Python 实现
import numpy as np
import cv2
sdThresh = 10
font = cv2.FONT_HERSHEY_SIMPLEX
def distMap(frame1, frame2):
"""outputs pythagorean distance between two frames"""
frame1_32 = np.float32(frame1)
frame2_32 = np.float32(frame2)
diff32 = frame1_32 - frame2_32
norm32 = np.sqrt(diff32[:,:,0]**2 + diff32[:,:,1]**2 + \
diff32[:,:,2]**2)/np.sqrt(255**2 + 255**2 + 255**2)
dist = np.uint8(norm32*255)
return dist
cv2.namedWindow('frame')
cv2.namedWindow('dist')
#capture video stream from camera source. 0 refers to first camera, 1 referes to 2nd and so on.
cap = cv2.VideoCapture(0)
_, frame1 = cap.read()
_, frame2 = cap.read()
facecount = 0