当前位置: 代码网 > it编程>前端脚本>Python > Python实现多路视频多窗口播放功能

Python实现多路视频多窗口播放功能

2025年02月10日 Python 我要评论
一、python实现多路视频播放功能服务端开发后通常需要做功能测试、性能测试,通常postman、curl等作功能测试使用,长跑服务性能postman并不太适合,如用c++实现播放器功能太慢,效率太低

一、python实现多路视频播放功能

服务端开发后通常需要做功能测试、性能测试,通常postman、curl等作功能测试使用,长跑服务性能postman并不太适合,如用c++实现播放器功能太慢,效率太低效,本文介绍一种用python来实现多路视频播放的测试。

二、代码实现

http申请视频流地址并cv2播放功能

import json
import requests
import time
import threading
from threading import thread
import signal
import base64
from io import bytesio
import queue
import random
import sys
from openpyxl import load_workbook
import json
import cv2
import datetime
import os
import shutil
import numpy as np
requests.packages.urllib3.disable_warnings()


group_id_base = 31000000452158000001
default_group = 31000000452168000002
username = "admin"
password = "admin123"
sever_ip = "1237.0.0.1"
my_token = "d21dcd7b-9380-cc90-7da4-673bde3bf2cf"

def alloctoken(sever_ip):
    url = '%s/micplatform/vuds/alloctoken' % (sever_ip)
    print(url)
    headers = {
        "content-type": "application/json",
    }
    data = {
        "validatemethod": "name+password",
        "username": str(username),
        "userpassword": str(password),
        "refreshinterval": 3600,
        "requestid": "1"
    }
    response = requests.post(url, headers=headers, verify=false, data=json.dumps(data))
    print(response.text)
    resp = json.loads(response.text)
    print( "url "+ url +" token: " + str(resp["token"]))
    return resp["token"]
def allocstream(deviceid):
    url = '%s/micplatform/vmd/realplayurlalloc' %(sever_ip)
    headers = {
        "content-type": "application/json",
    }
    data = {
        "requestid": "1",
        "token": my_token,
        "deviceid": deviceid,
        "protocol": "http+flv"
    }
    try:
        response = requests.post(url, headers=headers, verify=false, data=json.dumps(data))
        response.raise_for_status()  # check for http errors
        try:
            result = response.json()
            if "playurl" in result:
                print(f"{deviceid} 申请码流成功 {result['playurl']}")
                return result['playurl']
            else:
                print(f"{deviceid} 申请码流失败 {response.json()}" + my_token)
                return ""
        except json.jsondecodeerror:
            print(f"{deviceid} 响应解析失败:无法解析json")
            return ""
    except requests.requestexception as e:
        print(f"{deviceid} 请求失败:{e}")
        return ""

def freestream(playurl):
    url = '%s/micplatform/vmd/realplayurlrelease' %(sever_ip)
    headers = {
        "content-type": "application/json",
    }
    data = {
        "requestid": "1",
        "token": my_token,
        "playurl": playurl
    }
    try:
        response = requests.post(url, headers=headers, verify=false, data=json.dumps(data))
        response.raise_for_status()  # check for http errors
        try:
            result = response.json()
            if "resultdesc" in result and result['resultdesc'] == "成功":
                print(f"{playurl} 释放码流成功")
            else:
                print(f"{playurl} 申请码流失败")
        except json.jsondecodeerror:
            print(f"{playurl} 响应解析失败:无法解析json" )
    except requests.requestexception as e:
        print(f"{playurl} 请求失败:{e}")


def openvideo(streamtype,deviceid,stop_event):
    playurl = allocstream(deviceid)
    if len(playurl) == 0:
        return

    if len(playurl) != 0:
        playurl_array.append(playurl)
    if (streamtype == 3 or streamtype == 2):
        cap = cv2.videocapture(playurl)
    else:
        cap = cv2.videocapture(0)
    while (not stop_event.is_set()):
        ret, frame = cap.read()  # get a frame
        if ret == true:
            # showdate = str(datetime.datetime.now())
            # font = cv2.font_hershey_simplex
            # frame = cv2.puttext(frame, showdate, (10, 100), font, 0.5, (0, 255, 255), 2, cv2.line_aa)
            cv2.imshow(deviceid, frame)  # show a frame
            if cv2.waitkey(1) & 0xff == ord('q'):
                print("deviceid "+ deviceid + " receive the stop command")
                stop_event.set()  # 设置事件,通知其他线程停止
                break
        else:
            break
    cap.release()
    # 如果你的程序在退出时没有正确关闭所有opencv窗口,那么可能是因为cv2.destroyallwindows()没有在主线程中被调用。在所有线程结束后,确保在主线程中调用
    # cv2.destroyallwindows()

device_datas = []
stop_event = threading.event()  # 创建一个事件对象
def allocstreamtaskbyexcelfile(xlsfile):
    workbook = load_workbook(xlsfile)
    sheet = workbook.active
    for row in sheet.iter_rows(values_only=true):
        if len(row[0]) != 20 :
            continue
        if row[4] != "on":
            continue
        print("insert davice_data id: ",row[0])
        device_datas.append(row[0])
    process_array_in_threads(device_datas)

playurl_array = []
def process_array_in_threads(device_datas):
    print(device_datas)
    start_time = time.time()
    threads = []
    for deviceid in device_datas:
        thread = threading.thread(target=openvideo, args=(3,deviceid,stop_event))
        threads.append(thread)
        thread.start()
        print("start task stream "+ deviceid)
    for thread in threads:
        thread.join()

    cv2.destroyallwindows()
    for playurl in playurl_array:
        freestream(playurl)
    playurl_array.clear()

if __name__ == '__main__':
    try:
        file_path = "config.ini"
        with open(file_path, 'r') as f:
            config = json.load(f)

        if 'serverurl' in config:
            sever_ip = config['serverurl']
        else:
            sever_ip = ""
            print("please check serverurl fielddata in config.ini")

        if 'user' in config:
            username = config['user']
        else:
            username = ""
            print("please check username fielddata in config.ini")

        if 'password' in config:
            password = config['password']
        else:
            password = ""
            print("please check password fielddata in config.ini")

    except filenotfounderror:
        print(f"{file_path} does not exist")
        exit(0)
    except ioerror:
        print(f"{file_path} exists but is not readable")
        exit(0)
    except json.jsondecodeerror:
        print(f"{file_path} is not a valid json file")
        exit(0)

    # 配置文件格式检查
    if not sever_ip or not username or not password:
        print("config file is missing required fields. please check serverurl, user, password.")
        exit(0)

    print("*************************************")
    print("get server url in config.ini: " + sever_ip)
    print("get user in config.ini: " + username)
    print("get password in config.ini: " + password)
    print("*************************************")
    my_token = alloctoken(sever_ip)
    print(sever_ip + " get a token is  " + my_token)

    file = "name2id_vplatform.xlsx"
    allocstreamtaskbyexcelfile(file)

输入文件 name2id_vplatform.xlsx

配置文件输入参数:config.ini

三、打包代码实现

基于pycharm软件,安装打包软件

pip install pyinstaller
pyinstaller --onefile main.py
pyinstaller --onefile --distpath dist --out my_application.exe your_script.py

--onefile 表示创建一个独立的文件。
--distpath dist 指定输出目录为 dist。
--out my_application 指定输出的文件名为 my_application.exe。
your_script.py 是你想要打包的 python 脚本。

总结

本文实现了最简单最快的方式实现播放器功能,python实现视频播放多路实时流的视频。

到此这篇关于python实现多路视频多窗口播放功能的文章就介绍到这了,更多相关python多路视频播放内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!

(0)

相关文章:

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论

验证码:
Copyright © 2017-2025  代码网 保留所有权利. 粤ICP备2024248653号
站长QQ:2386932994 | 联系邮箱:2386932994@qq.com