Python远程视频监控程序的实例代码


Posted in Python onMay 05, 2019

老板由于事务繁忙无法经常亲临教研室,于是让我搞个监控系统,让他在办公室就能看到教研室来了多少人。o(>?<)o|||

最初我的想法是直接去网上下个软件,可是找来找去不是有毒就是收费,无奈技术不到家无法破解,只得另寻他法。

正当没有办法的时候,我看到一篇博文一个基于python的高速视频传输程序 ,看完茅塞顿开,觉得完全可以自己写一个,在此感谢作者詹姆斯。

这个程序包括一个服务器和一个客户端。需要的库有 VideoCapture 和 pygame,一个用来得到摄像头的视频,一个用来显示。Python库可以点这里下载:Python Extension Packages。进去后ctrl+F找到相应的库,然后选择相应的版本即可,这里还有很多其他的库可提供下载。

我想到的解决方案是,在教研室开一台电脑,接一个USB摄像头,然后开启一个服务器程序,等待着老板使用客户端连接,由于是实时视频传输,使用UDP协议。(主要传输部分采用詹姆斯的代码)。

服务器端代码如下:

# -*- coding: UTF-8 -*-
import socket
import time
import traceback
from VideoCapture import Device
import threading
# 全局变量
is_sending = False
cli_address = ('', 0)
# 主机地址和端口
host = ''
port = 10218
# 初始化UDP socket
ser_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
ser_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
ser_socket.bind((host, port))
# 接收线程类,用于接收客户端发送的消息
class UdpReceiver(threading.Thread):
 def __init__(self):
  threading.Thread.__init__(self)
  self.thread_stop = False
 def run(self):
  while not self.thread_stop:
   # 声明全局变量,接收消息后更改
   global cli_address 
   global is_sending
   try:
    message, address = ser_socket.recvfrom(2048)
   except:
    traceback.print_exc()
    continue
  #  print message,cli_address
   cli_address = address
   if message == 'startCam':
    print 'start camera',
    is_sending = True
    ser_socket.sendto('startRcv', cli_address)    
   if message == 'quitCam':
    is_sending = False
    print 'quit camera',
 def stop(self):
  self.thread_stop = True
# 创建接收线程
receiveThread = UdpReceiver()
receiveThread.setDaemon(True)   # 该选项设置后使得主线程退出后子线程同时退出
receiveThread.start()
# 初始化摄像头
cam = Device()
cam.setResolution(320,240)
# 主线程循环,发送视频数据
while 1:
 if is_sending:  
  img = cam.getImage().resize((160,120))
  data = img.tostring()
  ser_socket.sendto(data, cli_address) 
  time.sleep(0.05)
 else:
  time.sleep(1)
receiveThread.stop()
ser_socket.close()

服务器启动一个子线程,来监听客户端发送的消息。当有消息时,将is_sending改为True,则服务器向该客户端发送视频数据。具体信息可以看代码注释。

客户端代码如下:

# -*- coding: UTF-8 -*-
import socket, time
import pygame
from pygame.locals import *
from sys import exit
# 服务器地址,初始化socket
ser_address = ('localhost', 10218)
cli_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
# 设置超时
cli_socket.settimeout(5)
# 向服务器发送消息,并判断接收时是否超时,若超时则重发
while 1:
 cli_socket.sendto('startCam', ser_address)
 try:
  message, address = cli_socket.recvfrom(2048)
  if message == 'startRcv':
   print message
   break
 except socket.timeout:
  continue
# 此句无用。。防止窗口初始化后等待数据
cli_socket.recvfrom(65536)
# 初始化视频窗口
pygame.init()
screen = pygame.display.set_mode((640,480))
pygame.display.set_caption('Web Camera')
pygame.display.flip()
# 设置时间,可以用来控制帧率
clock = pygame.time.Clock()
# 主循环,显示视频信息
while 1:
 try:
  data, address = cli_socket.recvfrom(65536)
 except socket.timeout:
  continue
 camshot = pygame.image.frombuffer(data, (160,120), 'RGB')
 camshot = pygame.transform.scale(camshot, (640, 480))
 for event in pygame.event.get():
  if event.type == pygame.QUIT:
   cli_socket.sendto('quitCam', ser_address)
   cli_socket.close()
   pygame.quit()
   exit()
 screen.blit(camshot, (0,0))
 pygame.display.update() 
 clock.tick(20)

客户端就是简单地向服务器发送启动消息,接收到回复后开始进入主循环开始接收视频数据并显示。

由于UDP协议不保证信息是否成功到达,因此前面设置了个重发机制,只有当客户端收到服务器的回复后,才停止发送开启消息并进入主循环。具体见注释。

使用时将localhost改成服务器IP即可,目前测试仅适用于局域网,校园网。外网暂未测试,熟悉网络编程的同学可以自行实验。

经验

调试的时候出现过服务器怎么都收不到客户端消息,结果调试一下午都找不到原因。晚上回来把防火墙、安全软件全关了,顺利通过。

服务器开启新线程后,由于Python奇怪的设定,主线程退出后子线程得完成后才会退出,而这里子线程又是一个死循环,因此需要对子线程调用setDaemon(True),这样主线程退出时子线程也会自动退出。若没有调用该方法,调试一次后第二次可能失败,因为后台还有个子线程在运行。

Python 相关文章推荐
Python中除法使用的注意事项
Aug 21 Python
Python类方法__init__和__del__构造、析构过程分析
Mar 06 Python
Python中字典的基本知识初步介绍
May 21 Python
Python中asyncore异步模块的用法及实现httpclient的实例
Jun 28 Python
你所不知道的Python奇技淫巧13招【实用】
Dec 14 Python
python3实现SMTP发送邮件详细教程
Jun 19 Python
Python标准库使用OrderedDict类的实例讲解
Feb 14 Python
Django框架创建项目的方法入门教程
Nov 04 Python
pytorch 求网络模型参数实例
Dec 30 Python
python数据库编程 ODBC方式实现通讯录
Mar 27 Python
Django实现微信小程序支付的示例代码
Sep 03 Python
Python tkinter之ComboBox(下拉框)的使用简介
Feb 05 Python
Python统计一个字符串中每个字符出现了多少次的方法【字符串转换为列表再统计】
May 05 #Python
20行python代码实现人脸识别
May 05 #Python
使用python实现mqtt的发布和订阅
May 05 #Python
Python向excel中写入数据的方法
May 05 #Python
python使用pandas处理大数据节省内存技巧(推荐)
May 05 #Python
python使用MQTT给硬件传输图片的实现方法
May 05 #Python
Python实现的插入排序,冒泡排序,快速排序,选择排序算法示例
May 04 #Python
You might like
PHP数组 为文章加关键字连接 文章内容自动加链接
2011/12/29 PHP
php中限制ip段访问、禁止ip提交表单的代码分享
2014/08/22 PHP
php使用$_POST或$_SESSION[]向js函数传参
2014/09/16 PHP
php画图实例
2014/11/05 PHP
PHP生成json和xml类型接口数据格式
2015/05/17 PHP
PHP封装的简单连接MongoDB类示例
2019/02/13 PHP
兼容Mozilla必须知道的知识。
2007/01/09 Javascript
jQuery Tools Dateinput使用介绍
2012/07/14 Javascript
js函数的延迟加载实现代码
2012/10/11 Javascript
解决JS浮点数运算出现Bug的方法
2013/03/12 Javascript
探讨在JQuery和Js中,如何让ajax执行完后再继续往下执行
2013/07/09 Javascript
引入JS文件IE6报语法错误或缺少对象问题的解决方法
2014/01/09 Javascript
JS交换变量的方法
2015/01/21 Javascript
微信小程序 定义全局数据、函数复用、模版等详细介绍
2016/10/27 Javascript
微信小程序 图片边框解决方法
2017/01/16 Javascript
JS实现移动端判断上拉和下滑功能
2017/08/07 Javascript
Vue中自定义全局组件的实现方法
2017/12/08 Javascript
使用JS获取SessionStorage的值
2018/01/12 Javascript
vue2.0+koa2+mongodb实现注册登录
2018/04/10 Javascript
详解vue 不同环境配置不同的打包命令
2019/04/07 Javascript
bootstrap Table实现合并相同行
2019/07/19 Javascript
JS 事件机制完整示例分析
2020/01/15 Javascript
[01:32]2016国际邀请赛中国区预选赛IG战队首日赛后采访
2016/06/27 DOTA
[01:09:16]DOTA2-DPC中国联赛 正赛 SAG vs Dynasty BO3 第一场 1月25日
2021/03/11 DOTA
python:pandas合并csv文件的方法(图书数据集成)
2018/04/12 Python
简单了解python高阶函数map/reduce
2019/06/28 Python
python 模拟贷款卡号生成规则过程解析
2019/08/30 Python
Python3将数据保存为txt文件的方法
2019/09/12 Python
Python求解正态分布置信区间教程
2019/11/20 Python
python跨文件使用全局变量的实现
2020/11/17 Python
css3选择器基本介绍
2014/12/15 HTML / CSS
是什么让J2EE适合用来开发多层的分布式的应用
2015/01/16 面试题
教师实习自我鉴定
2013/12/14 职场文书
地理信息科学专业推荐信
2014/09/08 职场文书
司法局群众路线教育实践活动开展情况总结
2014/10/25 职场文书
分享CSS盒子模型隐藏的几种方式
2022/02/28 HTML / CSS