python视频转码脚本

今天有一个临时的需求,就是需要将一个wmv的初步转码成mp4的格式。找了一圈,免费的工具少,即使有免费的工具,在功能上也是有所限制,或者会给你塞广告或者附带安装其它流氓小游戏或者杀毒程序。

我并非不支持正版,我本人不是专业的视频创作者,平日里对视频转码编辑的需求极少,为这临时需要而买个正常软件,一没必要,二不划算。

FFmpeg 是一个强大的开源视频工具包,通过命令行可以轻易的将一个 wmv 文件转码成需要的目标格式文件,例如转码成mp4

Python 也是个很好用的脚本语言。为可以快速的搜索视频文档,然后对其进行转码操作,使用Python写了一个简单的脚本,该脚本可以借助Everything的强大搜索功能,协助用户快速的搜索定位需要转码的文档,然后对其进行转码。

Python 脚本

Python脚本比较简单,逻辑也比较清晰。截图如下:

20240511141928

如👆上图所示,脚本首先定义了一个命令行入参的类,用于管理和解析命令行传入的参数。
然后定义了一个小的函数,这个函数并不是核心功能,仅用于对比两个指定的字符串中的差异。

在主程序中(if __name__ == '__main__'部分),脚本首先解析了来自命令行的参数;
然后定义了一个Everything搜索接口对象(该Everything搜索接口对象提供了基于Everything的搜索功能),用于协助用户快速的搜索定位兴趣文档。

在功能逻辑循环体中,首先引导用户搜索定位待转码的视频文档;然后与用户交互确定转码的目标格式;最后借助 moviepy这个Python包完成视频转码操作。

以上,脚本逻辑清晰自然,贴出Python代码如下👇。

# -*- coding:UTF-8 -*-
"""
@author: dyy
@contact: douyaoyuan@126.com
@time: 2024/5/10 21:40
@file: 视频转码.py
@desc: xxxxxx
"""# region 引入必要的依赖
import os模块名 = 'DebugInfo'
try:from DebugInfo.DebugInfo import *
except ImportError as impErr:print(f"尝试导入 {模块名} 依赖时检测到异常:{impErr}")print(f"尝试安装 {模块名} 模块:")try:os.system(f"pip install {模块名}")except OSError as osErr:print(f"尝试安装模块 {模块名} 时检测到异常:{osErr}")exit(0)else:try:from DebugInfo.DebugInfo import *except ImportError as impErr:print(f"再次尝试导入 {模块名} 依赖时检测到异常:{impErr}")exit(0)模块名 = 'difflib'
try:import difflib  # 需要安装 difflib 模块,以支持字符差异对比操作
except ImportError as impErr:print(f"尝试导入 {模块名} 依赖时检测到异常:{impErr}")print(f"尝试安装 {模块名} 模块:")try:os.system(f"pip install {模块名}")except OSError as osErr:print(f"尝试安装模块 {模块名} 时检测到异常:{osErr}")exit(0)else:try:import difflibexcept ImportError as impErr:print(f"再次尝试导入 {模块名} 依赖时检测到异常:{impErr}")exit(0)模块名 = 'moviepy'
try:from moviepy.editor import VideoFileClip  # 需要引入 moviepy 模块,以支持视频转码的操作
except ImportError as impErr:print(f"尝试导入 {模块名} 依赖时检测到异常:{impErr}")print(f"尝试安装 {模块名} 模块:")try:os.system(f"pip install {模块名}")except OSError as osErr:print(f"尝试安装模块 {模块名} 时检测到异常:{osErr}")exit(0)else:try:from moviepy.editor import VideoFileClipexcept ImportError as impErr:print(f"再次尝试导入 {模块名} 依赖时检测到异常:{impErr}")exit(0)
# endregion# 定义一个 命令行参数类,用于解析和记录命令行参数
class 命令行参数类(入参基类):def __init__(self):super().__init__()self._添加参数('搜索关键字', str, '指定文档搜索的关键字')self._添加参数('everything服务端口', str, 'everything HTTP 服务端口', '22')# region 访问器@propertydef jsonCfg(self) -> str:if 'jsonCfg' in self._参数字典:return self._参数字典['jsonCfg'].else:return ''@jsonCfg.setterdef jsonCfg(self,: str):if 'jsonCfg' in self._参数字典:self._参数字典['jsonCfg'].= str()@propertydef 搜索关键字(self) -> str:if '搜索关键字' in self._参数字典:return self._参数字典['搜索关键字'].else:return ''@搜索关键字.setterdef 搜索关键字(self,: str):if '搜索关键字' in self._参数字典:self._参数字典['搜索关键字'].= str()@propertydef everything服务端口(self) -> str:if 'everything服务端口' in self._参数字典:return self._参数字典['everything服务端口'].else:return ''@everything服务端口.setterdef everything服务端口(self,: str):if 'everything服务端口' in self._参数字典:self._参数字典['everything服务端口'].= str()# endregion# 标注两个字符串的差异项
def 差异标注(字串1: str, 字串2: str) -> tuple[str, str]:字串1 = str(字串1 if 字串1 else '').strip()字串2 = str(字串2 if 字串2 else '').strip()# region 原文档变动对齐# 使用SequenceMatcher()类计算两个字符串的相似度匹配器 = difflib.SequenceMatcher(None, 字串1, 字串2)差异 = 匹配器.get_opcodes()# 差异标注字串1差异标注: str = ''字串2差异标注: str = ''for opcode, i1, i2, j1, j2 in 差异:if opcode == 'equal':字串1差异标注 = 字串1差异标注 + 字串1[i1:i2]字串2差异标注 = 字串2差异标注 + 字串2[j1:j2]elif opcode == 'delete':字串1差异标注 = 字串1差异标注 + 红字(字串1[i1:i2])elif opcode == 'insert':字串2差异标注 = 字串2差异标注 + 绿字(字串2[j1:j2])elif opcode == 'replace':字串1差异标注 = 字串1差异标注 + 红字(字串1[i1:i2])字串2差异标注 = 字串2差异标注 + 绿字(字串2[j1:j2])return 字串1差异标注, 字串2差异标注if __name__ == '__main__':画板 = 打印模板(False)画板.执行位置(__file__)# region 解析入参入参 = 命令行参数类()入参.解析Json(画板=画板.副本.缩进())入参.解析入参(画板=画板.副本.缩进())if 画板.正在调试:入参.展示(画板=画板.副本.缩进())搜索关键字: str = 入参.搜索关键字 if 入参.搜索关键字 else ''# endregion# region 定义搜索接口本地搜索: 搜索接口类 = 搜索接口类()if 在nt系统中():if 入参.everything服务端口:本地搜索.everything服务端口 = int(入参.everything服务端口)else:画板.提示错误('入参.everything服务端口 参数无效')exit(1)# endregion# region 开始逻辑循环while True:def 文档排除规则(文档: str) -> bool:  # 排除一些不必要的干扰文档if not 文档:  # 空文档不要return Trueif '\\recent\\' in 文档.lower():  # recent 文档夹内的文档不要return Trueif 文档.lower().endswith('.lnk'):  # *.lnk 文档不要return True# region 引导用户选择需要转码的视频文档while True:原视频文档列表 = 交互接口类.指定选择文档(输入提示='请输入关键字以定位视频文档(0: 退出程序):',搜索接口=本地搜索,搜索关键字=搜索关键字,候选项上限=250,排除规则=文档排除规则,多选=True,画板=画板.副本.缩进())if '0' in 原视频文档列表:# 用户要求退出程序exit(0)if 原视频文档列表:画板.消息('您选择的文档列出如下:')for 文档 in 原视频文档列表:画板.消息(绿字(文档))break# endregion# region 目标格式信息交互目标视频文档列表: list[str] = []while True:目标格式后缀名: str = 交互接口类.发起文本交互(输入提示='请输入视频目标格式(0: 退出程序):',画板=画板.副本.缩进())if '0' == 目标格式后缀名:# 用户要求退出程序exit(0)if 目标格式后缀名.startswith('。'):目标格式后缀名 = f'.{目标格式后缀名[1:]}'if not 目标格式后缀名.startswith('.'):目标格式后缀名 = f'.{目标格式后缀名}'if not 目标格式后缀名.strip('.'):# 如果后缀名是空的,则重新要求输入continue# 后缀名统一转换成小写格式目标格式后缀名 = 目标格式后缀名.lower()# 合成目标文档名:for 文档 in 原视频文档列表:目标视频文档列表.append(f'{文档}{目标格式后缀名}')# 打的并提示用户确认转换方式画板.消息(黄字('即将启动的视频转码如下:'))画板.准备表格(对齐控制串='rl')for 序号 in range(len(原视频文档列表)):原文档,目标文档 = 差异标注(原视频文档列表[序号],目标视频文档列表[序号])画板.添加一行(原文档,洋红字('->'),目标文档)画板.展示表格(列间距=[0,0])转码确认: str = 交互接口类.发起文本交互(输入提示='是否确认转码?(y: 确认; n: 修改目标格式; 0: 退出程序)',限定范围='YyNn0',画板=画板.副本.缩进())if '0' == 转码确认:# 用户要求退出exit(0)elif 'y' == 转码确认.lower():# 用户确认转码break# endregion# region 开始视频转码编解码格式: str = ''for 序号 in range(len(原视频文档列表)):视频切片 = VideoFileClip(原视频文档列表[序号])while True:if not 编解码格式:编解码格式 = 交互接口类.发起文本交互(输入提示=f'默认 codec 参数是 libx264, 输入 {绿字("y")} 以确认; 'f'或者您可以输入其它编码格式({红字("0")}: {红字("退出程序")}):',画板=画板.副本.缩进())if '0' == 编解码格式:# 用户要求退出程序exit(0)编解码格式 = 编解码格式.lower()if 'y' == 编解码格式:# 用户确认使用 libx264 作为 codec 参数编解码格式 = 'libx264'try:if 目标格式后缀名.endswith('gif'):视频切片.write_gif(filename=目标视频文档列表[序号])elif 编解码格式:视频切片.write_videofile(filename=目标视频文档列表[序号], codec=编解码格式)else:视频切片.write_videofile(filename=目标视频文档列表[序号])except Exception as exp:画板.提示错误('视频转码遇到异常如下:')画板.消息(exp.__str__())# 清除 codec 参数,重新发起交互编解码格式 = ''continueelse:# 如果转码顺利完成,则关闭 视频切片,开始下一段视频的处理视频切片.close()break# endregion# endregion

使用演示

👉第一步,脚本首先引导用户搜索和定位视频文件,我们可以输入视频文件名信息以定位视频文档,然后参过序号选择目标文档;如下👇:
20240511144403

👉第二步,脚本询问我们需要将视频文档转换为什么样的目标格式,我们输入目标文档的后缀名,然后再一次确认脚本的问询;如下👇:
20240511144652
在这一步中,如果我们输入了错误的目标格式,我们还可以输入 n 来重新修改目标格式。

👉第三步,脚本会询问我们是否使用默认的 codec 参数(默认为libx264)进行视频转码,如果我们希望使用其它 codec 参数,例如 png,或者libvorbis 等,我们也可以直接输入 codec 值来告诉脚本;如下👇:
Snipaste_2024-05-11_14-51-48
Snipaste_2024-05-11_14-51-52
以上👆脚本交互完成后,脚本即开始了视频转码过程,我们耐心等待转码完成即可,如下👇:
20240511145705

懒人包

以上Python脚本的功能比较简单,对于视频转码的参数控制较少,但如果大家有需要,可以轻易的加入更多的转码参数控制功能。

以上 Python脚本,对应的bat调用脚本,和对应的cfg参数文档,统一进行了打包,见:Python 视频转码脚本。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/324971.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

前端笔记-day05

文章目录 01-结构伪类选择器02-结构伪类选择器-公式用法03-伪元素选择器04-盒子模型-组成05-盒子模型-边框线06-盒子模型-单方向边框线07-盒子模型-内边距08-盒子模型-padding多值写法09-盒子模型-尺寸计算10-盒子模型-版心居中11-清除默认样式12-元素溢出overflow13-外边距合并…

贪心算法-----柠檬水找零

今日题目:leetcode860 题目链接:点击跳转题目 分析: 顾客只会给三种面值:5、10、20,先分类讨论 当收到5美元时:不用找零,面值5张数1当收到10美元时:找零5美元,面值5张数…

3588 pwm android12 的操作

问题: 客户需要在android12 的界面上操作板卡上的 PWM 蜂鸣器设备。 过程: 1 了解一下 3588 android12 源码的 关于PWM 的驱动。 设备树找不到 pwm 但是, 还不知道,android12 最终包含的 设备树是哪个,但是经过我的…

Meilisearch使用过程趟过的坑

Elasticsearch 做为老牌搜索引擎,功能基本满足,但复杂,重量级,适合大数据量。 MeiliSearch 设计目标针对数据在 500GB 左右的搜索需求,极快,单文件,超轻量。 所以,对于中小型项目来说…

使用html和css实现个人简历表单的制作

根据下列要求,做出下图所示的个人简历(表单) 表单要求 Ⅰ、表格整体的边框为1像素,单元格间距为0,表格中前六列列宽均为100像素,第七列 为200像素,表格整体在页面上居中显示; Ⅱ、前…

2024年电工杯数学建模竞赛A题B题思路代码分享

您的点赞收藏是我继续更新的最大动力! 一定要点击如下的卡片链接,那是获取资料的入口! 点击链接加入群聊【2024电工杯】:http://qm.qq.com/cgi-bin/qm/qr?_wv1027&k_PrjarulWZU8JsAOA9gnj_oHKIjFe195&authKeySbv2XM853…

简洁大气APP下载单页源码

源码介绍 简洁大气APP下载单页源码,源码由HTMLCSSJS组成,记事本打开源码文件可以进行内容文字之类的修改,双击html文件可以本地运行效果,也可以上传到服务器里面 效果截图 源码下载 简洁大气APP下载单页源码

3D Web轻量化引擎HOOPS Communicator如何处理DWG文件中的图纸?

在当今工程设计和建筑领域,数字化技术已经成为不可或缺的一部分。HOOPS Communicator作为一种强大的三维数据可视化工具,被广泛应用于处理各种CAD文件,其中包括AutoCAD的DWG格式。在这篇文章中,我们将探讨HOOPS Communicator是如何…

【Win10设备管理器中无端口选项】

计算机疑难杂症分享002 Win10设备管理器中无端口选项1、问题现象2、问题原因3、问题解决3.1、驱动精灵(亲测的此方法)3.2、添加过时硬件3.3、官方的方法 Win10设备管理器中无端口选项 1、问题现象 当我调试串口通信时,发现打开设备管理器没有端口,打开…

Centos7安装图形化界面

前言:原文在我的博客网站中,持续更新数通、系统方面的知识,欢迎来访! Centos7安装图形化界面https://myweb.myskillstree.cn/43.html 目录 一、安装GNOME桌面 二、开机自启动修改为命令行模式 三、卸载图形化界面 一、安装GN…

重生奇迹mu再生宝石怎么用有什么用

重生奇迹mu再生宝石有2个用处: 1、在玛雅哥布林处给380装备加PVP属性4追4以上的380级装备,守护宝石一颗,再生宝石一颗,成功得到PVP装备,失败宝石消失,装备无变化; 2、给非套装点强化属性用法跟祝福,灵魂,生命一样直接往装备上敲,成功得到随机强化属性一…

conan2 基础入门(05)-(静态库动态库)(DebugRelease)

conan2 基础入门(05)-(静态库&动态库)(Debug&Release) 文章目录 conan2 基础入门(05)-(静态库&动态库)(Debug&Release)⭐准备预备文件和Code ⭐静态库&动态库静态库动态库 ⭐Debug&ReleaseReleaseDebug END视频教学settings.yml ⭐准备 本文均在windo…

设计模式-创建型-原型模式-prototype

工作经验类 public class WorkExperience implements Cloneable {private String workDate;private String company;public void setWorkDate(String workDate) {this.workDate workDate;}public void setCompany(String company) {this.company company;}Overridepublic Ob…

数据结构与算法学习笔记三---循环队列的表示和实现(C++)

目录 前言 1.为什么要使用循环队列 2.队列的顺序存储方式的实现 1.定义 2.队列初始化 3.销毁 4.清空队列 5.队列是否为空 6.队列长度 7.队头 8.入队 9.出队 10.遍历队列 11.完整代码 3.参考资料 前言 这篇文章介绍循环队列的表示和用法。 1.为什么要使用循环队…

Vue3:路由

1. 路由简介 在Vue3中,路由是一个核心概念,特别是在构建单页面应用程序(SPA)时。以下是Vue3中路由的基本概念: 1. **路由(Route)**:在Vue3中,路由是指根据特定的规则将用…

Leetcode3138. 同位字符串连接的最小长度

Every day a Leetcode 题目来源:3138. 同位字符串连接的最小长度 解法1:枚举同位子串的长度 从小到大枚举字符串 t 的长度 len。 因为字符串 s 由字符串 t 和它的同位字符串连接而成,所以 n % len 0。 然后比较所有首字母下标为 0、len…

HTML/CSS3

1.CSS CSS的作用在于在HTML的基础上(决定网页的内容和结构)对网页进行排版布局 对网页中的元素提供样式 使得网页显得更加精美CSS全称是cascading style sheets 即层叠样式表CSS样式的书写格式:样式名: 样式值 例如:color: red建议:之后进行空格 CSS样式…

redis的双写一致性

双写一致性问题 1.先删除缓存或者先修改数据库都可能出现脏数据。 2.删除两次缓存,可以在一定程度上降低脏数据的出现。 3.延时是因为数据库一般采用主从分离,读写分离。延迟一会是让主节点把数据同步到从节点。 1.读写锁保证数据的强一致性 因为一般放…

在 Python 的哪个版本之后,字典的添加顺序与键的顺序是一致的?

🍉 CSDN 叶庭云:https://yetingyun.blog.csdn.net/ 在 Python 的不同版本中,字典(dict)类型的行为发生了显著变化。在 Python 3.6 及之前的版本中,字典是无序的,这意味着字典在遍历时不能保证按…

笨方法自学python(一)

我觉得python和c语言有很多相似之处,如果有c语言基础的话学习python也不是很难。这一系列主要是学习例题来学习python;我用的python版本是3.12 代码编辑器我用的是notepad,运行py程序用cmd 现在开始写第一个程序: print ("…