Wan2.2完整部署实战:从零搭建个人视频生成平台
2025/12/25 10:04:28
核心使用opencv-python(OpenCV)库处理视频和图像,这是Python视觉处理的主流库,先通过pip安装:
# 安装opencv-pythonpipinstallopencv-python# 若需处理特殊视频格式(如MP4编码问题),可额外安装ffmpeg(可选)# Windows:下载ffmpeg并配置环境变量;Ubuntu:sudo apt install ffmpeg;Mac:brew install ffmpeg适合大多数标注场景,可灵活设置提取间隔,避免图片过多冗余。
importcv2importosdefvideo_to_images_basic(video_path,# 输入视频文件路径(如:test.mp4)output_dir,# 输出图片文件夹路径frame_interval=30,# 提取间隔:每30帧提取1张(默认30帧,对应1秒/帧,可调整)img_format="jpg",# 输出图片格式(jpg/png,推荐jpg,占用空间小)img_prefix="frame"# 图片文件名前缀(如:frame_0001.jpg)):# 1. 创建输出文件夹(若不存在则自动创建)ifnotos.path.exists(output_dir):os.makedirs(output_dir)print(f"创建输出文件夹:{output_dir}")# 2. 打开视频文件cap=cv2.VideoCapture(video_path)ifnotcap.isOpened():raiseValueError(f"无法打开视频文件:{video_path}")# 3. 获取视频基本信息total_frames=int(cap.get(cv2.CAP_PROP_FRAME_COUNT))# 视频总帧数fps=cap.get(cv2.CAP_PROP_FPS)# 视频帧率(每秒帧数)print(f"视频信息:总帧数={total_frames},帧率={fps:.2f},预计提取图片数={total_frames//frame_interval+1}")# 4. 循环提取帧并保存frame_count=0# 当前帧计数器saved_count=0# 已保存图片计数器whileTrue:# 读取一帧视频ret,frame=cap.read()# 若读取失败(已到视频末尾),退出循环ifnotret:break# 按间隔提取帧ifframe_count%frame_interval==0:# 构造图片文件名(补零对齐,方便排序)img_name=f"{img_prefix}_{saved_count:04d}.{img_format}"img_path=os.path.join(output_dir,img_name)# 保存图片cv2.imwrite(img_path,frame)print(f"已保存:{img_path}")saved_count+=1frame_count+=1# 5. 释放资源cap.release()print(f"\n转换完成!共保存{saved_count}张图片,存储路径:{output_dir}")# ------------------- 调用示例 -------------------if__name__=="__main__":# 配置参数VIDEO_PATH="input_video.mp4"# 你的视频文件路径OUTPUT_DIR="extracted_images"# 图片输出文件夹FRAME_INTERVAL=30# 每30帧提取1张(1秒1张,若视频帧率25,则每1.2秒1张)# 执行转换video_to_images_basic(video_path=VIDEO_PATH,output_dir=OUTPUT_DIR,frame_interval=FRAME_INTERVAL,img_format="jpg")适合需要按固定时间间隔(如每2秒提取1张)的场景,比按帧间隔更直观,不受视频帧率影响。
importcv2importosdefvideo_to_images_by_time(video_path,# 输入视频文件路径output_dir,# 输出图片文件夹路径time_interval=1.0,# 时间间隔:每1.0秒提取1张(可调整,如2.0=每2秒1张)img_format="jpg",# 输出图片格式img_prefix="frame"# 图片文件名前缀):# 1. 创建输出文件夹ifnotos.path.exists(output_dir):os.makedirs(output_dir)print(f"创建输出文件夹:{output_dir}")# 2. 打开视频文件cap=cv2.VideoCapture(video_path)ifnotcap.isOpened():raiseValueError(f"无法打开视频文件:{video_path}")# 3. 获取视频基本信息total_frames=int(cap.get(cv2.CAP_PROP_FRAME_COUNT))fps=cap.get(cv2.CAP_PROP_FPS)video_duration=total_frames/fps# 视频总时长(秒)frame_interval=int(fps*time_interval)# 换算为帧间隔print(f"视频信息:总帧数={total_frames},帧率={fps:.2f},总时长={video_duration:.2f}秒")print(f"时间间隔={time_interval}秒,对应帧间隔={frame_interval}帧,预计提取图片数={int(video_duration//time_interval)+1}")# 4. 循环提取帧并保存frame_count=0saved_count=0last_saved_time=0.0# 上一次保存图片的时间whileTrue:ret,frame=cap.read()ifnotret:break# 获取当前帧对应的时间(秒)current_time=frame_count/fps# 按时间间隔保存ifcurrent_time-last_saved_time>=time_interval:img_name=f"{img_prefix}_{saved_count:04d}.{img_format}"img_path=os.path.join(output_dir,img_name)cv2.imwrite(img_path,frame)print(f"时间{current_time:.2f}秒:已保存{img_path}")saved_count+=1last_saved_time=current_time frame_count+=1# 5. 释放资源cap.release()print(f"\n转换完成!共保存{saved_count}张图片,存储路径:{output_dir}")# ------------------- 调用示例 -------------------if__name__=="__main__":VIDEO_PATH="input_video.mp4"OUTPUT_DIR="extracted_images_by_time"TIME_INTERVAL=2.0# 每2秒提取1张图片video_to_images_by_time(video_path=VIDEO_PATH,output_dir=OUTPUT_DIR,time_interval=TIME_INTERVAL,img_format="jpg")cv2.VideoCapture(video_path):打开视频文件,返回视频捕获对象;cap.read():读取一帧视频,返回(ret, frame),ret为布尔值(是否读取成功),frame为当前帧图像(numpy数组);cap.get(cv2.CAP_PROP_FRAME_COUNT):获取视频总帧数;cap.get(cv2.CAP_PROP_FPS):获取视频帧率(每秒播放的帧数);cv2.imwrite(img_path, frame):将帧图像保存为图片文件;frame_interval(方案1):帧间隔,数值越大提取图片越少,推荐30(对应1秒1张);time_interval(方案2):时间间隔,按需设置(如0.5=每0.5秒1张,5.0=每5秒1张)。frame_0001.jpg,方便后续标注软件按顺序加载;jpg(占用空间小)和png(无损压缩,适合高精度标注);cap.release()释放视频资源,避免内存泄漏。"D:/videos/test.mp4",(Linux/Mac):"/home/user/videos/test.mp4";ffmpeg -i input.avi -c:v libx264 output.mp4);转换后的图片可使用以下主流标注工具进行标注: