Python基于OpenCV将视频逐帧保存为图片

Python基于OpenCV将视频逐帧保存为图片

# coding=utf-8
"""
原文地址:https://blog.csdn.net/bskfnvjtlyzmv867/article/details/79970146

原博主的代码不能在我的数据集上直接运行所以进行了稍微的修改

注意:路径下有中文时cv2.imwrite()函数无法保存(至少在我这是这样为了保险起见最好不要有中文)

"""
import os
import cv2

videos_src_path = "D:\\GraduationProject\\data\\train\\"
# video_formats = [".MP4", ".MOV"]          我的数据集都是.mp4所以不需要进行分类判断
frames_save_path = "D:\\GraduationProject\\image_data\\"
width = 720
height = 480
time_interval = 10


def video2frame(video_src_path, frame_save_path, frame_width, frame_height, interval):
    """
    将视频按固定间隔读取写入图片
    :param video_src_path: 视频存放路径
    :param formats: 包含的所有视频格式
    :param frame_save_path: 保存路径
    :param frame_width: 保存帧宽
    :param frame_height: 保存帧高
    :param interval: 保存帧间隔
    :return: 帧图片
    """
    videos = os.listdir(video_src_path)

    # def filter_format(x, all_formats):      无需判断所以这部分代码也就不需要了
    #     if x[-4:] in all_formats:
    #         return True
    #     else:
    #         return False
    #
    # videos = filter(lambda x: filter_format(x, formats), videos)

    for each_video in videos:
        # print "正在读取视频:", each_video
        print("正在读取视频:", each_video)    # 我的是Python3.6

        each_video_name = each_video[:-4]
        os.mkdir(frame_save_path + each_video_name)
        each_video_save_full_path = os.path.join(frame_save_path, each_video_name) + "/"

        each_video_full_path = os.path.join(video_src_path, each_video)

        cap = cv2.VideoCapture(each_video_full_path)
        frame_index = 0
        frame_count = 0
        if cap.isOpened():
            success = True
        else:
            success = False
            print("读取失败!")

        while(success):
            success, frame = cap.read()
           # print "---> 正在读取第%d帧:" % frame_index, success
            print("---> 正在读取第%d帧:" % frame_index, success)      # 我的是Python3.6

            if frame_index % interval == 0 and success:     # 如路径下有多个视频文件时视频最后一帧报错因此条件语句中加and success
                resize_frame = cv2.resize(frame, (frame_width, frame_height), interpolation=cv2.INTER_AREA)
                # cv2.imwrite(each_video_save_full_path + each_video_name + "_%d.jpg" % frame_index, resize_frame)
                cv2.imwrite(each_video_save_full_path + "%d.jpg" % frame_count, resize_frame)
                frame_count += 1

            frame_index += 1

    	cap.release()		# 这行要缩一下、原博客会报错

def main():
    video2frame(videos_src_path, frames_save_path, width, height, time_interval)
if __name__ == '__main__':
    main()

原文博客:https://blog.csdn.net/bskfnvjtlyzmv867/article/details/79970146

  • 5
    点赞
  • 54
    收藏
    觉得还不错? 一键收藏
  • 9
    评论
好的,以下是一个基于 PythonOpenCV 线程播放视频、抽帧显示并发送到云端检测的案例,供您参考: ```python import cv2 import threading import queue import time import boto3 # AWS S3 配置信息 s3 = boto3.client('s3', aws_access_key_id='your_access_key_id', aws_secret_access_key='your_secret_access_key') bucket_name = 'your_bucket_name' # 视频抽帧线程类 class FrameThread(threading.Thread): def __init__(self, queue, video_path, frame_rate): threading.Thread.__init__(self) self.queue = queue self.video_path = video_path self.frame_rate = frame_rate def run(self): cap = cv2.VideoCapture(self.video_path) frame_count = int(cap.get(cv2.CAP_PROP_FRAME_COUNT)) i = 0 while cap.isOpened(): ret, frame = cap.read() if ret: if i % self.frame_rate == 0: self.queue.put(frame) i += 1 else: break cap.release() # 图片检测线程类 class DetectThread(threading.Thread): def __init__(self, queue, s3, bucket_name): threading.Thread.__init__(self) self.queue = queue self.s3 = s3 self.bucket_name = bucket_name def run(self): while True: if not self.queue.empty(): frame = self.queue.get() # TODO: 在此处进行图片检测,并将检测结果存储到云端 # detect_result = detect(frame) # s3.put_object(Bucket=self.bucket_name, Body=detect_result, Key='result.jpg') time.sleep(0.1) # 主函数 if __name__ == '__main__': video_path = 'your_video_path' frame_rate = 10 # 抽帧率 queue = queue.Queue() # 启动视频抽帧线程 frame_thread = FrameThread(queue, video_path, frame_rate) frame_thread.start() # 启动图片检测线程 detect_thread = DetectThread(queue, s3, bucket_name) detect_thread.start() # 开始播放视频并实时显示抽取的帧 cap = cv2.VideoCapture(video_path) while cap.isOpened(): ret, frame = cap.read() if ret: cv2.imshow('frame', frame) if cv2.waitKey(1) & 0xFF == ord('q'): break else: break cap.release() cv2.destroyAllWindows() ``` 在上述代码中,我们定义了两个线程类:`FrameThread` 和 `DetectThread`,分别用于视频抽帧和图片检测。在 `FrameThread` 中,我们使用 OpenCV 库读取视频文件,并按照设定的抽帧率将抽取的帧存储到一个队列中。在 `DetectThread` 中,我们从队列中取出帧进行图片检测,并将检测结果保存到云端。 在主线程中,我们首先启动 `FrameThread` 和 `DetectThread`,然后使用 OpenCV 库播放视频,并实时显示抽取的帧。当用户按下 `q` 键时,程序停止播放视频并退出。 请注意:在实际使用时,您需要根据自己的需求进行修改和完善,如替换视频路径、AWS 账号信息、图片检测算法等。同时,由于图片检测可能会比较耗时,所以在 `DetectThread` 中我们使用了 `time.sleep(0.1)` 进行简单的降频处理,以免队列中的帧积压过多导致内存溢出。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 9
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值