opencv读取rtsp的一些优化

这段代码展示了如何使用Python的queue模块和多线程将视频同步读取转换为异步处理。在Receive函数中,从rtsp流中捕获帧并放入队列,Display函数则从队列中取出帧进行显示。此外,还对比了OpenCV VideoCapture的read()和grab()方法在处理速度上的差异,并探讨了多解码器同时解码的效率。
部署运行你感兴趣的模型镜像

用队列将同步转为异步

import cv2
import queue
import time
import threading
q=queue.Queue()
 
def Receive():
    print("start Reveive")
    cap = cv2.VideoCapture("rtsp://admin:admin_123@172.0.0.0")
    ret, frame = cap.read()
    q.put(frame)
    while ret:
        ret, frame = cap.read()
        q.put(frame)
 
 
def Display():
     print("Start Displaying")
     while True:
         if q.empty() !=True:
            frame=q.get()
            cv2.imshow("frame1", frame)
         if cv2.waitKey(1) & 0xFF == ord('q'):
                break
 
if __name__=='__main__':
    p1=threading.Thread(target=Receive)
    p2 = threading.Thread(target=Display)
    p1.start()
    p2.start()

抓取与跳帧

import cv2
import queue
import time
import threading
q=queue.Queue()
 
def Receive():
    print("start Reveive")
    cap = cv2.VideoCapture("rtsp://admin:admin_123@172.0.0.0")
    ret, frame = cap.read()
    path = ""
    size = (int(vidcap.get(cv2.CAP_PROP_FRAME_WIDTH)),int(vidcap.get(cv2.CAP_PROP_FRAME_HEIGHT)))   
    fps = 25
    out_video = cv2.VideoWriter(path, cv2.VideoWriter_fourcc('m','j','p','g'), fps, size)
    q.put(frame)
    fp = 0
    while ret:
    	fp += 1
        ret, frame = cap.read()
        out_video.write(frame)
        if (fp % 2 == 0):
        	q.put(frame)
 
 
def Display():
     print("Start Displaying")
     while True:
         if q.empty() !=True:
            frame=q.get()
            cv2.imshow("frame1", frame)
         if cv2.waitKey(1) & 0xFF == ord('q'):
                break
 
if __name__=='__main__':
    p1=threading.Thread(target=Receive)
    p2 = threading.Thread(target=Display)
    p1.start()
    p2.start()
"""
read() and grab() in OpenCV VideoCapture takes different time as there is no
decoding for grab() operation. Let's try to find out how much these two methods
differs.
"""
import cv2


def main():
    tm = cv2.TickMeter()

    # =========================================================================
    # Test 1, grab and decode each frame.

    cap = cv2.VideoCapture()
    cap.open("/home/robin/Videos/clip1-ssd.mp4", cv2.CAP_FFMPEG)
    total_frame_count = cap.get(cv2.CAP_PROP_FRAME_COUNT)

    tm.start()
    # print("Total frames number: {}".format(total_frame_count))
    f_count = 0
    while True:
        f_count += 1
        r, img = cap.read()
        if r == False:
            break
        # for _ in range(24):
        #     cap.grab()
    tm.stop()
    print("Read and decode {} frames takes {} seconds.".format(
        f_count, tm.getTimeSec()/tm.getCounter()))

    # =========================================================================
    # Test 2, grab only without decoding.

    cap = cv2.VideoCapture()
    cap.open("/home/robin/Videos/clip1-ssd.mp4", cv2.CAP_FFMPEG)
    total_frame_count = cap.get(cv2.CAP_PROP_FRAME_COUNT)

    tm.reset()
    tm.start()
    while True:
        r = cap.grab()
        if r == False:
            break
    tm.stop()
    print("Read {} frames takes {} seconds.".format(
        total_frame_count, tm.getTimeSec()/tm.getCounter()))

    # =========================================================================
    # Test 3, reading same file with two decoders. This block failed on my PC.
    # Maybe it's not supported by FFMPEG?

    from multiprocessing.pool import ThreadPool
    thread_num = cv2.getNumberOfCPUs()
    pool = ThreadPool(processes=thread_num)

    def get_frame_at(src, location):
        cap = cv2.VideoCapture()
        cap.open(src, cv2.CAP_FFMPEG)
        cap.set(cv2.CAP_PROP_FRAME_COUNT, location)
        rst, frame = cap.read()
        return rst, frame

    def get_frame_next(cap):
        rst, frame = cap.read()
        return rst, frame

    video_src = "/home/robin/Videos/clip1-ssd.mp4"
    cap1 = cv2.VideoCapture()
    cap1.open(video_src, cv2.CAP_FFMPEG)
    cap2 = cv2.VideoCapture()
    cap2.open(video_src, cv2.CAP_FFMPEG)

    tm.reset()
    tm.start()
    f_count = 0
    while True:
        f_count += 1
        t1 = pool.apply_async(get_frame_next, (cap1, ))
        t2 = pool.apply_async(get_frame_next, (cap2, ))
        print(t1.get()[0])
        if (t1.get()[0] is False) and (t2.get()[0] is False):
            break

    tm.stop()
    print("Decode same file with 2 decoder takes {} seconds.".format(
        tm.getTimeSec()/tm.getCounter()))


if __name__ == "__main__":
    main()
然后清楚了grab的含义,我们就可以对开始的

    	fp += 1
        success = cap.grab()
        if (fp % 2 == 0):
        	continue
        _,image = cap.retrive()

参考文献:

  • https://www.icode9.com/content-4-816043.html

您可能感兴趣的与本文相关的镜像

Python3.11

Python3.11

Conda
Python

Python 是一种高级、解释型、通用的编程语言,以其简洁易读的语法而闻名,适用于广泛的应用,包括Web开发、数据分析、人工智能和自动化脚本

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值