opencv python多线程视频捕获

时间:2013-04-03 13:03:18

标签: python multithreading opencv

我正在尝试阅读2个视频文件,并在单独的Windows中同时显示它们。 这是我的代码:

import threading
import cv2
threadLock=threading.Lock()
class myThread (threading.Thread):
    maxRetries=20
    def __init__(self, threadID, name,video_url):
        threading.Thread.__init__(self)
        self.threadID = threadID
        self.name = name
        self.video_url=video_url

    def attemptRead(self,cvVideo):
        threadLock.acquire()
        (isRead,cvImage)=cvVideo.read()
        threadLock.release()
        if isRead==False:
            count=1
            while isRead==False and count<myThread.maxRetries:
                threadLock.acquire()
                (isRead,cvImage)=cvVideo.read()
                threadLock.release()
                print self.name+' try no: ',count
                count+=1
        return (isRead,cvImage)

    def run(self):
        print "Starting " + self.name
        windowName = self.name
        cv2.namedWindow(windowName)
        cvVideo = cv2.VideoCapture(self.video_url)


        while True:
            (isRead,cvImage)=self.attemptRead(cvVideo)
            if isRead==False:
                break
            cv2.imshow(windowName,cvImage)
            key=cv2.waitKey(50)
            if key==27:
                break

        cv2.destroyWindow(windowName)
        print self.name + "Exiting"

def main():
    thread1 = myThread(1, "Thread1",'C:/Traffic Pics/Videos/Panjim Capture.mp4')
    thread2 = myThread(2, "Thread2",'C:/Traffic Pics/Videos/Miramar Capture.mp4')

    thread1.start()
    thread2.start()

print "Exiting Main Thread"

if __name__ == '__main__':
    main()

正在发生的是 主题2 窗口正在显示。尝试阅读 max_retries限制之后的视频线程1退出(在我的情况下为10)。 问题是虽然我创建了单独的 cvVideo 对象,但我似乎无法同时使用它们。 可能是什么问题?

1 个答案:

答案 0 :(得分:4)

*编辑:我将保留下面的代码,但我猜您有编解码器问题?我安装了xvid编解码器(这是Megamind.avi编码的示例),并且该程序可以正常运行megamind视频中的一个或两个线程。你能让megamind视频在单线程版本中运行吗?

如果有帮助,这里是an SO post on opencv video codecs。这是我使用的xvid download(k-lite对我不起作用)。


你写的代码基本上适合我。对于您和其他想要尝试的人,我做了以下事情:

  • 已应用PEP 8建议
  • 删除了不必要的读取尝试代码(在工作视频的最后一帧之后也删除了误导性线程重新读取警告)
  • 使用了一个视频文件,也许每个人都使用opencv
  • 至少在我的系统上删除了对opencv似乎无关紧要的线程锁定
  • 移动了一些其他的小东西

VideoCapture.read可能会出现其他错误,这会使read_attempt方法变得有价值,但我只能找到the two errors the docs mention。对于那些它只返回代码已经测试过的假。

import os
import threading

import cv2

my_opencv_path = "C:/opencv2.4.3"
video_path_1 = os.path.join(my_opencv_path, "samples", "cpp", "tutorial_code",
                            "HighGUI", "video-input-psnr-ssim", "video",
                            "Megamind.avi")
video_path_2 = os.path.join(my_opencv_path, "samples", "c", "tree.avi")
assert os.path.isfile(video_path_1)
assert os.path.isfile(video_path_2)


class MyThread (threading.Thread):
    maxRetries = 20

    def __init__(self, thread_id, name, video_url, thread_lock):
        threading.Thread.__init__(self)
        self.thread_id = thread_id
        self.name = name
        self.video_url = video_url
        self.thread_lock = thread_lock

    def run(self):
        print "Starting " + self.name
        window_name = self.name
        cv2.namedWindow(window_name)
        video = cv2.VideoCapture(self.video_url)
        while True:
            # self.thread_lock.acquire()  # These didn't seem necessary
            got_a_frame, image = video.read()
            # self.thread_lock.release()
            if not got_a_frame:  # error on video source or last frame finished
                break
            cv2.imshow(window_name, image)
            key = cv2.waitKey(50)
            if key == 27:
                break
        cv2.destroyWindow(window_name)
        print self.name + " Exiting"


def main():
    thread_lock = threading.Lock()
    thread1 = MyThread(1, "Thread 1", video_path_1, thread_lock)
    thread2 = MyThread(2, "Thread 2", video_path_2, thread_lock)
    thread1.start()
    thread2.start()
    print "Exiting Main Thread"

if __name__ == '__main__':
    main()