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 getopt模块处理命令行选项实例
May 13 Python
Python用GET方法上传文件
Mar 10 Python
提升Python程序运行效率的6个方法
Mar 31 Python
python变量不能以数字打头详解
Jul 06 Python
python+selenium+autoit实现文件上传功能
Aug 23 Python
MySQL适配器PyMySQL详解
Sep 20 Python
numpy中实现ndarray数组返回符合特定条件的索引方法
Apr 17 Python
python实现多张图片拼接成大图
Jan 15 Python
如何在django中实现分页功能
Apr 22 Python
Python内存泄漏和内存溢出的解决方案
Sep 26 Python
Pycharm在指定目录下生成文件和删除文件的实现
Dec 28 Python
如何在C++中调用Python
May 21 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 DOMElement 操作xml 文档的实现代码
2013/05/10 PHP
PHP跨平台获取服务器IP地址自定义函数分享
2014/12/29 PHP
PHP函数nl2br()与自定义函数nl2p()换行用法分析
2016/04/02 PHP
PHP读书笔记整理_结构语句详解
2016/07/01 PHP
PHPExcel在linux环境下导出报500错误的解决方法
2017/01/26 PHP
PHP简单计算两个时间差的方法示例
2017/06/20 PHP
tp5.1 实现setInc字段自动加1
2019/10/18 PHP
在laravel中实现ORM模型使用第二个数据库设置
2019/10/24 PHP
JavaScript 异步调用框架 (Part 2 - 用例设计)
2009/08/03 Javascript
javascript面向对象之定义成员方法实例分析
2015/01/13 Javascript
jquery实现用户信息修改验证输入方法汇总
2015/07/18 Javascript
分享JavaScript与Java中MD5使用两个例子
2015/12/23 Javascript
分享10个优化代码的CSS和JavaScript工具
2016/05/11 Javascript
jquery轮播的实现方式 附完整实例
2016/07/28 Javascript
jquery mobile移动端幻灯片滑动切换效果
2020/04/15 Javascript
vue使用ajax获取后台数据进行显示的示例
2018/08/09 Javascript
[01:20:06]TNC vs VG 2018国际邀请赛小组赛BO2 第二场 8.16
2018/08/17 DOTA
使用setup.py安装python包和卸载python包的方法
2013/11/27 Python
python爬虫教程之爬取百度贴吧并下载的示例
2014/03/07 Python
初学Python函数的笔记整理
2015/04/07 Python
Python中不同进制的语法及转换方法分析
2016/07/27 Python
Python3 翻转二叉树的实现
2019/09/30 Python
解决pymysql cursor.fetchall() 获取不到数据的问题
2020/05/15 Python
详解Windows下PyCharm安装Numpy包及无法安装问题解决方案
2020/06/18 Python
HTML5的hidden属性兼容老浏览器的方法
2014/04/23 HTML / CSS
夏洛特和乔治婴儿和儿童时装精品店:Charlotte and George
2018/06/06 全球购物
世界上最好的旅行夹克:BauBax
2018/12/23 全球购物
Ever New美国:澳大利亚领先的女装时尚品牌
2019/11/28 全球购物
小学二年级评语
2014/04/21 职场文书
工作业绩不及格检讨书
2014/10/28 职场文书
优秀党员个人总结
2015/02/14 职场文书
党校团干班培训心得体会
2016/01/06 职场文书
仅仅使用 HTML/CSS 实现各类进度条的方式汇总
2021/11/11 HTML / CSS
postgreSQL数据库基础知识介绍
2022/04/12 PostgreSQL
Linux中一对多配置日志服务器的详细步骤
2022/07/23 Servers
Mysql如何查看是否使用到索引
2022/12/24 MySQL