Python计算机视觉编程 第十章

2024-09-08 08:36

本文主要是介绍Python计算机视觉编程 第十章,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

一、OpenCv基础知识

1.读取和写入图像

2.颜色空间

3.显示图像和结果

二、处理视频

1.输入视频

2.将视频读取到NumPy数组中

三、跟踪

1.光流

2.Lucas-Kanade算法


一、OpenCv基础知识

OpenCV 自带读取、写入图像函数以及矩阵操作和数学库。

1.读取和写入图像

import cv2
# 读取图像
im = cv2.imread(r"D:\test\pil.png")
h,w = im.shape[:2]
print (h,w)
# 保存图像
cv2.imwrite('result.png',im)

使用该代码可以读取一张图片,并且得到图片的尺寸信息。imwrite() 会根据文件后缀自动转换图像。

2.颜色空间

OpenCV里面颜色是BGR顺序存储的,读取时默认为BGR,可以使用cvColor来进行转换。其中的参数可以设置为:

cv2.COLOR_BGR2GRAY
cv2.COLOR_BGR2RGB
cv2.COLOR_GRAY2BGR

这些参数可以从字面上直接理解。

3.显示图像和结果

首先是一个简单的例子

import cv2
# 读取图像
im = cv2.imread(r"D:\test\pil.png")
gray = cv2.cvtColor(im,cv2.COLOR_BGR2GRAY)
# 计算积分图像
intim = cv2.integral(gray)
# 归一化并保存
intim = (255.0*intim) / intim.max()
cv2.imwrite('result.jpg',intim)

得到的结果为

上述代码将一幅图片保存为了一个整数图像。

之后是一个从一个种子像素进行泛洪填充的例子:

import cv2
import numpy as np
# 读取图像
print(cv2.__version__)
im = cv2.imread(r"D:\test\pil.png")
h,w = im.shape[:2]
# 泛洪填充
diff = (6,6,6)
mask = np.zeros((h+2,w+2),np.uint8)
cv2.floodfill(im,mask,(10,10), (255,255,0),diff,diff)
# 在 OpenCV 窗口中显示结果
cv2.imshow('flood fill',im)
cv2.waitKey()
# 保存结果
cv2.imwrite('result.jpg',im)

运行代码出现如下的错误:

最后一个例子为SURF特征提取, SURF 特征是 SIFT 特征的一个更快特征提取版,其实现代码为:

import cv2
import numpy as np
# 读取图像im = cv2.imread(r"D:\test\pil.png")
im_lowres = cv2.pyrDown(im)
# 变换成灰度图像
gray = cv2.cvtColor(im_lowres,cv2.COLOR_RGB2GRAY)
# 检测特征点
s = cv2.xfeatures2d.SIFT_create()
mask = np.uint8(np.ones(gray.shape))
keypoints = s.detect(gray,mask)
# 显示结果及特征点
vis = cv2.cvtColor(gray,cv2.COLOR_GRAY2BGR)
for k in keypoints[::10]:cv2.circle(vis,(int(k.pt[0]),int(k.pt[1])),2,(0,255,0),-1)cv2.circle(vis,(int(k.pt[0]),int(k.pt[1])),int(k.size),(0,255,0),2)
cv2.imshow('local descriptors',vis)
cv2.waitKey()

同样报了和上面错误类似得错误:

 网上查询之后发现是因为cv2.SURF()的问题,需要将其修改为cv2.xfeatures2d.SIFT_create(),之后再次执行就可以运行了。但是对于前一个例子的错误还是无法解决 。

最终得到的结果显示了图片的SURF特征。

二、处理视频

1.输入视频

OpenCV 能够很好地支持从摄像头读取视频。一个捕获视频帧并在 OpenCV 窗口中显示这些视频帧的完整例子为:

import cv2
# 设置视频捕获
cap = cv2.VideoCapture(0)
while True:ret,im = cap.read()cv2.imshow('video test',im)key = cv2.waitKey(10)if key == 27:breakif key == ord(' '):cv2.imwrite('vid_result.jpg',im)

得到的效果为:

其中的read() 方法解码并返回下一视频帧,第一个变量 ret 是一个判断视频帧是否成功读入,第二个变量是实际读入的图像数组。函数 waitKey() 等待用户按键:如果按下的是 Esc 键键,则退出应用;如果按下的是空格键,就保存该视频帧。

其次也可以使用给 GaussianBlur() 函数将图像模糊,这个函数会用高斯滤波器对传入的该帧图像进行滤波。

得到的结果为:

出来可以使用摄像头读取图像之外,还可以从文件里面读取视频帧, 再调用VideoCapture()函数是需要设置为如下的格式:

capture = cv2.VideoCapture('filename')

2.将视频读取到NumPy数组中

一个从摄像头捕获视频并将视频帧存储在一个 NumPy 数组中的例子为:

import cv2
import numpy as np
# 设置视频捕获
cap = cv2.VideoCapture(0)
frames = []
# 获取帧,存储到数组中
while True:ret,im = cap.read()cv2.imshow('video',im)frames.append(im)if cv2.waitKey(10) == 27:break
frames = np.array(frames)
# 检查尺寸
print (im.shape)
print (frames.shape)

当关闭打开的视频窗口之后,会在控制台输出一个视频帧的帧数、 帧高、帧宽和颜色通道数。

三、跟踪

 跟踪指的是在图像序列或视频里对其中的目标进行跟踪的过程。

1.光流

光流是目标、场景或摄像机在连续两帧图像间运动时造成的目标的运动。它是图像在平移过程中的二维矢量场。其主要依赖于三个假设:

  1. 亮度恒定:图像中目标的像素强度在连续帧之间不会发生变化
  2. 时间规律:相邻帧之间的时间足够短,以至于在考虑运行变化时可以忽略它们之间的差异
  3. 空间一致性:相邻像素具有相似的运动

对于相邻帧间的小运动以及短时间跳跃是一个很好的模型。假设一个目标像素在 t 时刻亮度为 I(x,y,t),那么它的光流方程为:

\nabla I^T\boldsymbol{v}=-I_{t}

OpenCV 中包含了一些光流实现的函数:

用了块匹配的 CalcOpticalFlowBM()

用了文献的 CalcOpticalFlowHS()

空间金字塔 Lucas-Kanade 算法 calcOpticalFlowPyrLK()

calcOpticalFlowFarneback()

其中最后一种是获取密集流场最好的方法之一,使用该方法的一个例子为:

import cv2
from numpy import *def draw_flow(im, flow, step=16):""" 在间隔分开的像素采样点处绘制光流 """h,w = im.shape[:2]y,x = mgrid[step/2:h:step, step/2:w:step].reshape(2, -1).astype('int64')fx, fy = flow[y, x].T# 创建线的终点lines = vstack([x, y, x + fx, y + fy]).T.reshape(-1, 2, 2)lines = int32(lines)# 创建图像并绘制vis = cv2.cvtColor(im, cv2.COLOR_GRAY2BGR)for (x1, y1), (x2, y2) in lines:cv2.line(vis, (x1, y1), (x2, y2), (0, 255, 0), 1)cv2.circle(vis, (x1, y1), 1, (0, 255, 0), -1)return vis# 设置视频捕获
cap = cv2.VideoCapture(0)
ret, im = cap.read()
prev_gray = cv2.cvtColor(im, cv2.COLOR_BGR2GRAY)while True:# 获取灰度图像ret, im = cap.read()gray = cv2.cvtColor(im, cv2.COLOR_BGR2GRAY)# 计算流flow = cv2.calcOpticalFlowFarneback(prev_gray, gray, None, 0.5, 3, 15, 3, 5, 1.2, 0)prev_gray = gray# 画出流失量cv2.imshow('Optical flow', draw_flow(gray, flow))if cv2.waitKey(10) == 27:break

运行了出现了如下的错误:

 将代码中错误前面的y和x类型修改为int类型即可:

上图为手从左向右移动产生的光流矢量。

2.Lucas-Kanade算法

跟踪最基本的形式是跟随感兴趣点,比如角点。其中一种算法是 Lucas-Kanade 跟踪算法,利用了稀疏光流算法。其可以应用于任何一种特征,不过通常使用一些角点。角点是结构张量中有两个较大特征值的那些点,且更小的特征值要大于某个阈值。

如果基于每一个像素考虑,该光流方程组是欠定方程,即每个方程中含很多未知变量。利用相邻像素有相同运动这一假设,可以将方程写为:

\begin{bmatrix}\nabla I^T(\mathbf{x}_1)\\\nabla I^T(\mathbf{x}_2)\\\vdots\\\nabla I^T(\mathbf{x}_n)\end{bmatrix}v=\begin{bmatrix}I_x(\mathbf{x}_1)&I_y(\mathbf{x}_1)\\I_x(\mathbf{x}_2)&I_y(\mathbf{x}_2)\\\vdots&\vdots\\I_x(\mathbf{x}_n)&I_y(\mathbf{x}_n)\end{bmatrix}\begin{bmatrix}\boldsymbol u\\\boldsymbol v\end{bmatrix}=-\begin{bmatrix}I_t(\mathbf{x}_1)\\I_t(\mathbf{x}_2)\\\vdots\\I_t(\mathbf{x}_n)\end{bmatrix}

可以使用最小二乘法对该方程进行求解。通过对于周围像素的贡献可以进行加权处理,使越远的像素影响越小,可以得到如下的关系:

\overline{\boldsymbol{M}}_lv=-\begin{bmatrix}I_t(\mathbf{x}_1)\\I_t(\mathbf{x}_2)\\\vdots\\I_t(\mathbf{x}_n)\end{bmatrix}, Or\; \; Av=b

对于该超定方程组也可以使用最小二乘法进行求解,从而得到运动矢量:

\boldsymbol{v}=(A^T\boldsymbol{A})^{-1}\boldsymbol{A}^T\boldsymbol{b}

其上为该跟踪算法运行矢量的求解过程,其具体实现为:

import cv2
# 一些常数及默认参数
lk_params = dict(winSize=(15,15),maxLevel=2,criteria=(cv2.TERM_CRITERIA_EPS | cv2.TERM_CRITERIA_COUNT,10,0.03))
subpix_params = dict(zeroZone=(-1,-1),winSize=(10,10),criteria = (cv2.TERM_CRITERIA_COUNT | cv2.TERM_CRITERIA_EPS,20,0.03))
feature_params = dict(maxCorners=500,qualityLevel=0.01,minDistance=10)
class LKTracker(object):def __init__(self,imnames):""" 使用图像名称列表初始化 """self.imnames = imnamesself.features = []self.tracks = []self.current_frame = 0

首先是需要建立一个跟踪类,用一个文件名列表对跟踪对象进行初始化,在利用一个变量对当前帧进行跟踪。之后就是需要载入实际图像,并转换成灰度图像,提取利用跟踪的好的特征点:

    def detect_points(self):self.image = cv2.imread(self.imnames[self.current_frame])self.gray = cv2.cvtColor(self.image,cv2.COLOR_BGR2GRAY)# 搜索好的特征点features = cv2.goodFeaturesToTrack(self.gray, **feature_params)# 提炼角点位置cv2.cornerSubPix(self.gray,features, **subpix_params)self.features = featuresself.tracks = [[p] for p in features.reshape((-1,2))]self.prev_gray = self.gray

之后需要获得下 一帧图像,然后应用 OpenCV 函数找出这些点运动到哪里 了,最后清除这些包含跟踪点的列表:

def track_points(self):""" 跟踪检测到的特征 """ if self.features != []:self.step() # 移到下一帧# 载入图像并创建灰度图像self.image = cv2.imread(self.imnames[self.current_frame])self.gray = cv2.cvtColor(self.image,cv2.COLOR_BGR2GRAY)#reshape() 操作,以适应输入格式tmp = np.float32(self.features).reshape(-1, 1, 2)# 计算光流features,status,track_error = cv2.calcOpticalFlowPyrLK(self.prev_gray,self.gray,tmp,None,**lk_params)# 去除丢失的点self.features = [p for (st,p) in zip(status,features) if st]features = np.array(features).reshape((-1,2))for i,f in enumerate(features):self.tracks[i].append(f)ndx = [i for (i,st) in enumerate(status) if not st]ndx.reverse()# 从后面移除for i in ndx:self.tracks.pop(i) self.prev_gray = self.graydef step(self,framenbr=None):""" 移到下一帧。如果没有给定参数,直接移到下一帧 """if framenbr is None:self.current_frame = (self.current_frame + 1) % len(self.imnames)else:self.current_frame = framenbr % len(self.imnames)

在真实场景中使用这个跟踪类的例子为:

from PIL import Image
from matplotlib import pyplot as plt
import lktrackimnames = [r"D:\test\images\bt.000.pgm", r"D:\test\images\bt.001.pgm", r"D:\test\images\bt.002.pgm", r"D:\test\images\bt.003.pgm"]
lkt = lktrack.LKTracker(imnames)
# 在第一帧进行检测,跟踪剩下的帧
img1 = Image.open(imnames[0])
img2 = Image.open(imnames[1])
img3 = Image.open(imnames[2])
img4 = Image.open(imnames[3])
plt.figure()
plt.subplot(141)
plt.imshow(img1)
plt.gray()
plt.axis('off')
plt.subplot(142)
plt.imshow(img2)
plt.gray()
plt.axis('off')
plt.subplot(143)
plt.imshow(img3)
plt.gray()
plt.axis('off')
plt.subplot(144)
plt.imshow(img4)
plt.gray()
plt.axis('off')
plt.show()
lkt.detect_points()
lkt.draw()
for i in range(len(imnames)-1):lkt.track_points()lkt.draw()

原始的4张图像为:

得到的结果为:

 之后还可以对其添加发生器:

    def track(self):for i in range(len(self.imnames)):if self.features == []:self.detect_points()else:self.track_points()# 创建一份 RGB 副本f = np.array(self.features).reshape(-1,2)im = cv2.cvtColor(self.image,cv2.COLOR_BGR2RGB)yield im,f

上面的函数可以使遍历整个序列并将获得的跟踪点和这些图像以 RGB 数组保存,以方便画出跟踪结果,之后可以画出这些点及这些点的跟踪结果:

from PIL import Image
from matplotlib import pyplot as plt
import lktrackimnames = [r"D:\test\dinosaur\viff.000.ppm", r"D:\test\dinosaur\viff.001.ppm", r"D:\test\dinosaur\viff.002.ppm", r"D:\test\dinosaur\viff.003.ppm"]
lkt = lktrack.LKTracker(imnames)
# 在第一帧进行检测,跟踪剩下的帧
img1 = Image.open(imnames[0])
img2 = Image.open(imnames[1])
img3 = Image.open(imnames[2])
img4 = Image.open(imnames[3])
plt.figure()
plt.subplot(141)
plt.imshow(img1)
plt.gray()
plt.axis('off')
plt.subplot(142)
plt.imshow(img2)
plt.gray()
plt.axis('off')
plt.subplot(143)
plt.imshow(img3)
plt.gray()
plt.axis('off')
plt.subplot(144)
plt.imshow(img4)
plt.gray()
plt.axis('off')
lkt = lktrack.LKTracker(imnames)
for im,ft in lkt.track():print ('tracking %d features' % len(ft))
# 画出轨迹
plt.figure()
plt.imshow(im)
for p in ft:plt.plot(p[0],p[1],'bo')
for t in lkt.tracks:plt.plot([p[0] for p in t],[p[1] for p in t])
plt.axis('off')
plt.show()

最终得到的结果为:

这篇关于Python计算机视觉编程 第十章的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1147629

相关文章

Python MySQL如何通过Binlog获取变更记录恢复数据

《PythonMySQL如何通过Binlog获取变更记录恢复数据》本文介绍了如何使用Python和pymysqlreplication库通过MySQL的二进制日志(Binlog)获取数据库的变更记录... 目录python mysql通过Binlog获取变更记录恢复数据1.安装pymysqlreplicat

利用Python编写一个简单的聊天机器人

《利用Python编写一个简单的聊天机器人》这篇文章主要为大家详细介绍了如何利用Python编写一个简单的聊天机器人,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 使用 python 编写一个简单的聊天机器人可以从最基础的逻辑开始,然后逐步加入更复杂的功能。这里我们将先实现一个简单的

基于Python开发电脑定时关机工具

《基于Python开发电脑定时关机工具》这篇文章主要为大家详细介绍了如何基于Python开发一个电脑定时关机工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 简介2. 运行效果3. 相关源码1. 简介这个程序就像一个“忠实的管家”,帮你按时关掉电脑,而且全程不需要你多做

Python实现高效地读写大型文件

《Python实现高效地读写大型文件》Python如何读写的是大型文件,有没有什么方法来提高效率呢,这篇文章就来和大家聊聊如何在Python中高效地读写大型文件,需要的可以了解下... 目录一、逐行读取大型文件二、分块读取大型文件三、使用 mmap 模块进行内存映射文件操作(适用于大文件)四、使用 pand

python实现pdf转word和excel的示例代码

《python实现pdf转word和excel的示例代码》本文主要介绍了python实现pdf转word和excel的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价... 目录一、引言二、python编程1,PDF转Word2,PDF转Excel三、前端页面效果展示总结一

Python xmltodict实现简化XML数据处理

《Pythonxmltodict实现简化XML数据处理》Python社区为提供了xmltodict库,它专为简化XML与Python数据结构的转换而设计,本文主要来为大家介绍一下如何使用xmltod... 目录一、引言二、XMLtodict介绍设计理念适用场景三、功能参数与属性1、parse函数2、unpa

Python中使用defaultdict和Counter的方法

《Python中使用defaultdict和Counter的方法》本文深入探讨了Python中的两个强大工具——defaultdict和Counter,并详细介绍了它们的工作原理、应用场景以及在实际编... 目录引言defaultdict的深入应用什么是defaultdictdefaultdict的工作原理

Python中@classmethod和@staticmethod的区别

《Python中@classmethod和@staticmethod的区别》本文主要介绍了Python中@classmethod和@staticmethod的区别,文中通过示例代码介绍的非常详细,对大... 目录1.@classmethod2.@staticmethod3.例子1.@classmethod

Python手搓邮件发送客户端

《Python手搓邮件发送客户端》这篇文章主要为大家详细介绍了如何使用Python手搓邮件发送客户端,支持发送邮件,附件,定时发送以及个性化邮件正文,感兴趣的可以了解下... 目录1. 简介2.主要功能2.1.邮件发送功能2.2.个性签名功能2.3.定时发送功能2. 4.附件管理2.5.配置加载功能2.6.

使用Python进行文件读写操作的基本方法

《使用Python进行文件读写操作的基本方法》今天的内容来介绍Python中进行文件读写操作的方法,这在学习Python时是必不可少的技术点,希望可以帮助到正在学习python的小伙伴,以下是Pyth... 目录一、文件读取:二、文件写入:三、文件追加:四、文件读写的二进制模式:五、使用 json 模块读写