使用Python和MoviePy库实现视频拼接与合成的技巧与实践

2024-05-09 13:44

本文主要是介绍使用Python和MoviePy库实现视频拼接与合成的技巧与实践,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

首先,我们需要导入所需的库:

 
from moviepy.editor import VideoFileClip, concatenate_videoclips, AudioFileClip, CompositeVideoClip, ColorClip
import os
import random
from typing import List
from enum import Enum

接下来,我们定义了两个枚举类,分别表示视频的长宽比例和视频拼接模式:

 
class VideoAspect(Enum):portrait = 1landscape = 2def to_resolution(self):if self == VideoAspect.portrait:return 1080, 1920elif self == VideoAspect.landscape:return 1920, 1080class VideoConcatMode(Enum):sequential = 1random = 2

然后,我们定义了一个名为combine_videos的函数,该函数接受以下参数:

  • combined_video_path:合成视频的输出路径。
  • video_paths:一个包含待拼接视频路径的列表。
  • audio_file:背景音乐文件的路径。
  • video_aspect:视频的长宽比例,默认为VideoAspect.portrait
  • video_concat_mode:视频拼接模式,默认为VideoConcatMode.random
  • max_clip_duration:每个视频片段的最大时长,默认为5秒。
  • threads:写入视频文件时使用的线程数,默认为2。

函数的主要逻辑如下:

  1. 加载音频文件并获取其时长。
  2. 计算每个视频片段的所需时长。
  3. 获取输出目录。
  4. 获取视频的分辨率。
  5. 循环添加视频片段直到达到音频的最大时长。
  6. 根据视频拼接模式决定视频片段的顺序。
  7. 遍历video_paths列表,对每个视频片段进行处理。
  8. 如果视频片段长于剩余的音频时长,则进行裁剪。
  9. 如果计算出的片段长度小于实际视频片段长度,则进行裁剪。
  10. 调整视频片段的大小以匹配目标分辨率。
  11. 如果视频片段的时长超过最大限制,则进行裁剪。
  12. 将处理后的视频片段添加到clips列表中。
  13. 合并所有视频片段。
  14. 将合成的视频写入文件。

最后,我们给出了一个示例,展示了如何使用这个函数来拼接视频:

 
combined_video_path = "output/combined_video.mp4"
video_paths = ["input/video1.mp4", "input/video2.mp4", "input/video3.mp4"]
audio_file = "input/audio.mp3"result = combine_videos(combined_video_path, video_paths, audio_file)
print(f"视频合成完成!输出文件路径: {result}")

完整代码如下:

from moviepy.editor import VideoFileClip, concatenate_videoclips, AudioFileClip, CompositeVideoClip, ColorClip
import os
import random
from typing import List
from enum import Enum# 定义视频的长宽比例
class VideoAspect(Enum):portrait = 1landscape = 2def to_resolution(self):if self == VideoAspect.portrait:return 1080, 1920elif self == VideoAspect.landscape:return 1920, 1080# 定义视频拼接模式
class VideoConcatMode(Enum):sequential = 1random = 2# 视频合成函数
def combine_videos(combined_video_path: str,video_paths: List[str],audio_file: str,video_aspect: VideoAspect = VideoAspect.portrait,video_concat_mode: VideoConcatMode = VideoConcatMode.random,max_clip_duration: int = 5,threads: int = 2) -> str:# 加载音频文件并获取其时长audio_clip = AudioFileClip(audio_file)audio_duration = audio_clip.durationprint(f"音频最大时长: {audio_duration} 秒")# 计算每个视频片段的所需时长req_dur = min(max_clip_duration, audio_duration / len(video_paths))print(f"每个片段的最大时长: {req_dur} 秒")# 获取输出目录output_dir = os.path.dirname(combined_video_path)# 获取视频的分辨率aspect = VideoAspect(video_aspect)video_width, video_height = aspect.to_resolution()clips = []video_duration = 0# 循环添加视频片段直到达到音频的最大时长while video_duration < audio_duration:# 根据视频拼接模式决定视频片段的顺序if video_concat_mode == VideoConcatMode.random:random.shuffle(video_paths)for video_path in video_paths:clip = VideoFileClip(video_path).without_audio()# 如果视频片段长于剩余的音频时长,则进行裁剪if (audio_duration - video_duration) < clip.duration:clip = clip.subclip(0, (audio_duration - video_duration))# 如果计算出的片段长度小于实际视频片段长度,则进行裁剪elif req_dur < clip.duration:clip = clip.subclip(0, req_dur)clip = clip.set_fps(30)# 调整视频片段的大小以匹配目标分辨率clip_w, clip_h = clip.sizeif clip_w != video_width or clip_h != video_height:clip_ratio = clip_w / clip_hvideo_ratio = video_width / video_height# 等比例缩放视频片段scale_factor = video_width / clip_w if clip_ratio > video_ratio else video_height / clip_hnew_width = int(clip_w * scale_factor)new_height = int(clip_h * scale_factor)clip_resized = clip.resize(newsize=(new_width, new_height))# 创建黑色背景以填充不匹配的部分background = ColorClip(size=(video_width, video_height), color=(0, 0, 0))clip = CompositeVideoClip([background.set_duration(clip.duration),clip_resized.set_position("center")])# 如果视频片段的时长超过最大限制,则进行裁剪if clip.duration > max_clip_duration:clip = clip.subclip(0, max_clip_duration)clips.append(clip)video_duration += clip.duration# 合并所有视频片段video_clip = concatenate_videoclips(clips)video_clip = video_clip.set_fps(30)print("正在写入视频文件...")# 将合成的视频写入文件video_clip.write_videofile(filename=combined_video_path,threads=threads,logger=None,temp_audiofile_path=output_dir,audio_codec="aac",fps=30)video_clip.close()print("视频合成完成!")return combined_video_path


【痕迹】QQ+微信朋友圈和聊天记录分析工具1.0.4
(1)纯Python语言实现,使用Flask后端,本地分析,不上传个人数据。

(2)数据可视化分析QQ、微信聊天记录,提取某一天的聊天记录与大模型对话。

  • 想不想知道经常和你聊天的那个人,聊天的频次,时段,内容云图?
  • 想不想知道你都在什么时候发朋友圈,都有谁给你点赞、评论?
  • 想不想知道哪一天的聊天频次最多,并把聊天记录甩给大模型,让它总结?


下载地址:https://www.lanzoub.com/b00rn0g47e 密码:9hww
 

这篇关于使用Python和MoviePy库实现视频拼接与合成的技巧与实践的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/973572

相关文章

python生成随机唯一id的几种实现方法

《python生成随机唯一id的几种实现方法》在Python中生成随机唯一ID有多种方法,根据不同的需求场景可以选择最适合的方案,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习... 目录方法 1:使用 UUID 模块(推荐)方法 2:使用 Secrets 模块(安全敏感场景)方法

一文详解如何使用Java获取PDF页面信息

《一文详解如何使用Java获取PDF页面信息》了解PDF页面属性是我们在处理文档、内容提取、打印设置或页面重组等任务时不可或缺的一环,下面我们就来看看如何使用Java语言获取这些信息吧... 目录引言一、安装和引入PDF处理库引入依赖二、获取 PDF 页数三、获取页面尺寸(宽高)四、获取页面旋转角度五、判断

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

MySQL 迁移至 Doris 最佳实践方案(最新整理)

《MySQL迁移至Doris最佳实践方案(最新整理)》本文将深入剖析三种经过实践验证的MySQL迁移至Doris的最佳方案,涵盖全量迁移、增量同步、混合迁移以及基于CDC(ChangeData... 目录一、China编程JDBC Catalog 联邦查询方案(适合跨库实时查询)1. 方案概述2. 环境要求3.

Spring StateMachine实现状态机使用示例详解

《SpringStateMachine实现状态机使用示例详解》本文介绍SpringStateMachine实现状态机的步骤,包括依赖导入、枚举定义、状态转移规则配置、上下文管理及服务调用示例,重点解... 目录什么是状态机使用示例什么是状态机状态机是计算机科学中的​​核心建模工具​​,用于描述对象在其生命

Spring Boot 结合 WxJava 实现文章上传微信公众号草稿箱与群发

《SpringBoot结合WxJava实现文章上传微信公众号草稿箱与群发》本文将详细介绍如何使用SpringBoot框架结合WxJava开发工具包,实现文章上传到微信公众号草稿箱以及群发功能,... 目录一、项目环境准备1.1 开发环境1.2 微信公众号准备二、Spring Boot 项目搭建2.1 创建

Linux进程CPU绑定优化与实践过程

《Linux进程CPU绑定优化与实践过程》Linux支持进程绑定至特定CPU核心,通过sched_setaffinity系统调用和taskset工具实现,优化缓存效率与上下文切换,提升多核计算性能,适... 目录1. 多核处理器及并行计算概念1.1 多核处理器架构概述1.2 并行计算的含义及重要性1.3 并

IntelliJ IDEA2025创建SpringBoot项目的实现步骤

《IntelliJIDEA2025创建SpringBoot项目的实现步骤》本文主要介绍了IntelliJIDEA2025创建SpringBoot项目的实现步骤,文中通过示例代码介绍的非常详细,对大家... 目录一、创建 Spring Boot 项目1. 新建项目2. 基础配置3. 选择依赖4. 生成项目5.

使用Python删除Excel中的行列和单元格示例详解

《使用Python删除Excel中的行列和单元格示例详解》在处理Excel数据时,删除不需要的行、列或单元格是一项常见且必要的操作,本文将使用Python脚本实现对Excel表格的高效自动化处理,感兴... 目录开发环境准备使用 python 删除 Excphpel 表格中的行删除特定行删除空白行删除含指定

全面掌握 SQL 中的 DATEDIFF函数及用法最佳实践

《全面掌握SQL中的DATEDIFF函数及用法最佳实践》本文解析DATEDIFF在不同数据库中的差异,强调其边界计算原理,探讨应用场景及陷阱,推荐根据需求选择TIMESTAMPDIFF或inte... 目录1. 核心概念:DATEDIFF 究竟在计算什么?2. 主流数据库中的 DATEDIFF 实现2.1