如何在视频中提取关键帧?

ops/2025/2/27 0:26:12/

视频处理中,提取关键帧是一项常见的任务。下面将介绍如何基于FFmpeg和Python,结合OpenCV库来实现从视频中提取关键帧的功能。

实现思路

  1. 使用FFmpeg获取视频的关键帧时间戳:FFmpeg是一个强大的视频处理工具,可以通过命令行获取视频中关键帧的时间戳信息。
  2. 使用OpenCV根据时间戳提取关键帧:OpenCV是一个广泛使用的计算机视觉库,可以根据视频的时间戳读取相应的帧。

代码实现

import cv2
import subprocess
import redef get_keyframe_timestamps(video_path):"""使用FFmpeg获取视频中关键帧的时间戳:param video_path: 视频文件的路径:return: 关键帧的时间戳列表"""# 构建FFmpeg命令command = ['ffmpeg','-i', video_path,'-skip_frame', 'nokey','-vsync', '0','-frame_pts', '1','-loglevel', 'verbose','-f', 'image2','-']try:# 执行FFmpeg命令并捕获输出result = subprocess.run(command, capture_output=True, text=True, check=True)output = result.stderr# 使用正则表达式提取关键帧的时间戳timestamps = re.findall(r'pts_time:([\d.]+)', output)timestamps = [float(ts) for ts in timestamps]return timestampsexcept subprocess.CalledProcessError as e:print(f"FFmpeg命令执行出错: {e.stderr}")return []def extract_keyframes(video_path, output_folder):"""根据关键帧的时间戳提取关键帧并保存到指定文件夹:param video_path: 视频文件的路径:param output_folder: 保存关键帧的文件夹路径"""# 打开视频文件cap = cv2.VideoCapture(video_path)if not cap.isOpened():print("无法打开视频文件")return# 获取关键帧的时间戳timestamps = get_keyframe_timestamps(video_path)for i, ts in enumerate(timestamps):# 设置视频的播放位置到关键帧的时间戳cap.set(cv2.CAP_PROP_POS_MSEC, ts * 1000)ret, frame = cap.read()if ret:# 保存关键帧到指定文件夹output_path = f"{output_folder}/keyframe_{i}.jpg"cv2.imwrite(output_path, frame)print(f"保存关键帧 {output_path}")# 释放视频捕获对象cap.release()if __name__ == "__main__":video_path = "your_video.mp4"  # 替换为实际的视频文件路径output_folder = "keyframes"  # 替换为实际的输出文件夹路径extract_keyframes(video_path, output_folder)

代码解释

  1. get_keyframe_timestamps函数

    • 使用FFmpeg命令获取视频中关键帧的时间戳信息。
    • 通过正则表达式从FFmpeg的输出中提取关键帧的时间戳。
  2. extract_keyframes函数

    • 打开视频文件。
    • 调用get_keyframe_timestamps函数获取关键帧的时间戳。
    • 根据时间戳设置视频的播放位置,并读取关键帧。
    • 将关键帧保存到指定的文件夹中。

使用方法

  1. 确保已经安装了FFmpeg和OpenCV库。
  2. 将代码中的video_path替换为实际的视频文件路径。
  3. 将代码中的output_folder替换为实际的输出文件夹路径。
  4. 运行代码,即可从视频中提取关键帧并保存到指定文件夹。

注意事项

  • 请确保FFmpeg的可执行文件已经添加到系统的环境变量中,否则可能会出现命令执行失败的问题。
  • 提取的关键帧将以keyframe_0.jpg, keyframe_1.jpg, … 的格式保存到指定的文件夹中。

http://www.ppmy.cn/ops/161545.html

相关文章

nvidia-docker2 和 NVIDIA Container Toolkit 的区别及推荐

NVIDIA Docker 和 NVIDIA Container Toolkit 1. NVIDIA Docker 和 NVIDIA Docker2 nvidia-docker 是 NVIDIA 最早推出的工具,用于在 Docker 容器中启用 GPU 支持。它以独立守护进程的形式作为 Volume Plugin 存在,但与 Docker 生态系统的兼容性较差&am…

【Linux】34.封装 UdpSocket(1)

文章目录 1. 实现一个简易的远程命令执行系统1.1 日志系统 (Log.hpp)1.2 UDP客户端 (UdpClient.cc)1.3 UDP服务器 (UdpServer.hpp)1.4 主程序 (main.c) 1. 实现一个简易的远程命令执行系统 1.1 日志系统 (Log.hpp) Log.hpp #pragma once // 防止头文件重复包含// 必要的头文…

DeepSeek掘金——SpringBoot 调用 DeepSeek API 快速实现应用开发

Spring Boot 实现 DeepSeek API 调用 1. 项目依赖 在 pom.xml 中添加以下依赖: <dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-webflux</artifactId></dependency>&l…

Maven之jjwt依赖爆红

在使用IDEA工具的时候&#xff0c;我们经常会出现pom.xml文件依赖速度慢或者是依赖不上等问题&#xff0c;导致pom.xml文件出现报错情况。如jjwt爆红 查了一下是阿里的库中没有这个依赖&#xff0c;索性直接使用jar包导入 第一步 下载jar包 打开Maven官网&#xff0c;寻找依…

HAProxy高级功能及配置

目录 1、HAProxy高级功能及配置 基于cookie的会话保持 配置选项 配置示例 验证cookie信息 HAProxy状态页 状态页配置项 启用状态页 登录状态页 backend server信息 利用状态页实现haproxy服务器的健康性检查 IP透传 layer 4 与 layer 7 四层负载 七层代理 四层…

白帽黑客系列教程之Windows驱动开发(64位环境)入门教程(六)

为什么要写这篇文章呢&#xff1f; 作为一名白帽黑客&#xff0c;如果想要学习ROOTKIT攻防技术&#xff0c;就必须要有能力进行驱动开发&#xff01; 本文章仅提供学习&#xff0c;切勿将其用于不法手段&#xff01; 在Windows操作系统的64位环境中&#xff0c;进行ROOTKIT攻…

Vue 不同大版本与 Node.js 版本匹配的详细参数

Vue 2.x 与 Node.js 版本匹配: Vue 2.x 细分版本建议 Node.js 版本理由支持状态2.0 - 2.610.x - 14.x这些 Node.js 版本在 Vue 2.x 早期开发和维护阶段广泛使用&#xff0c;能提供稳定运行环境&#xff0c;适配当时常用构建工具和依赖包Node.js 10.x 维护期已结束&#xff1b;…

在单位,领导不说,但自己得懂的7个道理

如果你感到很难继续适应旧模式、旧关系、旧想法&#xff0c;开始厌倦生活&#xff0c;你很可能到了该蜕皮的时候。把“不是自己”的那部分舍弃掉&#xff0c;你就能看见“自己是谁”了。 ——奥赞瓦罗尔&#xff0c;《读者》2024年第11期 前几天听部门里一个新来的小姑娘抱怨&a…