python调用ffmpeg命令行工具便捷操作视频示例实现过程
作者:Python 技术 发布时间:2023-12-19 07:48:22
文 | 李晓飞
来源:Python 技术「ID: pythonall」
最近有了一个新任务,需要将赛事视频,拆分成两分钟以内的小段,用于发布到短视频平台上。
本以为是个一次性的工作,结果赛事视频数据巨大,视频文件长短不一,完全没法手工处理,于是 Python 又一次拯救了我。
还等什么,开始干吧!
最重要的事
无论做什么事情,都要去分析一下最重要的是什么,然后集中精力攻克,再继续找最重要的事。
对我们这个任务来说,不算是个大项目,不过呢,还是要找最重要的事开始,步步为营,最终将整个问题解决了。
整体来来看,我们需要从一个目录中读取视频文件,然后,对每个视频文件进行裁剪,最后将处理好的文件保存好。
在这个过程中,最重要的是什么呢?我觉得,是视频裁剪,如果不能方便的裁剪视频,其他的一切工作都是白费的,是吧。
裁剪视频
现在短视频很流行,有很多视频编辑软件,功能丰富,而我们需要的只是裁剪功能,而且需要用编程的方式调用,那么最合适的莫过于 ffmpeg[1] 了。
ffmpeg 是一个命令行工具,功能强大,可以编程调用。
从 ffmpeg 官网上下载对应操作系统的版本,我下的是 Windows 版[2]。
下载后解压到一个目录,然后将目录下的 bin,配置到环境变量里。然后打开一个命令行,输入:
> ffmpeg -version
ffmpeg version 2021-10-07-git-b6aeee2d8b-full_build- ...
测试一下,能显示出版本信息,说明配置好了。
现在读一下文档,发现拆分视频文件的命令是:
ffmpeg -i [filename] -ss [starttime] -t [length] -c copy [newfilename]
]
i
为需要裁剪的文件
ss
为裁剪开始时间
t
为裁剪结束时间或者长度
c
为裁剪好的文件存放
好了,用 Python 写一个调用:
import subprocess as sp
def cut_video(filename, outfile, start, length=90):
cmd = "ffmpeg -i %s -ss %d -t %d -c copy %s" % (filename, start, length, outfile)
p = sp.Popen(cmd, shell=True)
p.wait()
return
定义了一个函数,通过参数传入
ffmpeg
需要的信息将裁剪命令写成一个字符串模板,将参数替换到其中
用
subprocess
的Popen
执行命令,其中参数shell=True
表示将命令作为一个整体执行p.wait()
很重要,因为裁剪需要一会儿,而且是另起进程执行的,所以需要等执行完成再做后续工作,否则可能找不到裁剪好的文件
这样视频裁剪工作就完成了,然后再看看什么是最重要的。
计算分段
视频裁剪时,需要一些参数,特别是开始时间,如何确定呢?如果这件事做不好,裁剪工作就很麻烦。
所以看看如何计算裁剪分段。
我需要将视频裁剪成一分半的小段,那么将需要知道目标视频文件的时间长度。
获取视频长度
如何获得长度呢?ffmpeg 提供了另一个命令 —— ffprobe
。
找了一下,可以合成一个命令来获取:
> ffprobe -v error -show_entries format=duration -of default=noprint_wrappers=1:nokey=1 -i a.flv
920.667
命令比较复杂哈,可以先不用管其他参数,只要将要分析的视频文件传入就好了。命令的结果是显示一行视频文件的长度。
于是可以编写一个函数:
import subprocess as sp
def get_video_duration(filename):
cmd = "ffprobe -v error -show_entries format=duration -of default=noprint_wrappers=1:nokey=1 -i %s" % filename
p = sp.Popen(cmd, stdout=sp.PIPE, stderr=sp.PIPE)
p.wait()
strout, strerr = p.communicate() # 去掉最后的回车
ret = strout.decode("utf-8").split("\n")[0]
return ret
函数只有一个参数,就是视频文件路径
合成命令语句,将视频文件路径替换进去
用
subprocess
来执行,注意这里需要设置一下命令执行后的输出用
wait
等待命令执行完成通过
communicate
提取输出结果从结果中提取视频文件的长度,返回
分段
得到了视频长度,确定好每个分段的长度,就可以计算出需要多少分段了。
代码很简单:
import math
duration = math.floor(float(get_video_duration(filename)))
part = math.ceil(duration / length)
注意,计算分段时,需要进行向上取整,即用 ceil
,以包含最后的一点尾巴。
得到了需要的分段数,用一个循环就可以计算出每一段的起始时间了。
获取文件
因为处理的文件很多,所以需要自动获取需要处理的文件。
方法很简单,也很常用,一般可以用 os.walk 递归获取文件,还可以自己写,具体根据实际情况。
for fname in os.listdir(dir):
fname = os.path.join(dir, os.path.join(dir, fname))
basenames = os.path.basename(fname).split('.')
mainname = basenames[0].split("_")[0]
...
提供视频文件所在的目录,通过 os.listdir
获取目录中的文件,然后,合成文件的绝对路径,因为调用裁剪命令时需要绝对路径比较方便。
获取文件名,是为了在后续对裁剪好的文件进行命名。
代码集成
现在每个部分都写好了,可以将代码集成起来了:
def main(dir):
outdir = os.path.join(dir, "output")
if not os.path.exists(outdir):
os.mkdir(outdir)
for fname in os.listdir(dir):
fname = os.path.join(dir, os.path.join(dir, fname))
if os.path.isfile(fname):
split_video(fname, outdir)
main
方法是集成后的方法先创建一个裁剪好的存储目录,放在视频文件目录中的 output 目录里
通过
listdir
获取到文件后,对每个文件进行处理,其中判断了一下是否为文件调用
split_video
方法开始对一个视频文件进行裁剪
来源:https://blog.csdn.net/weixin_48923393/article/details/120858140
猜你喜欢
- 如下所示:import numpynew_list = [i for i in range(9)]numpy.array(new_list)
- 步骤如下: 1.使用torchvision加载并预处理CIFAR-10数据集、2.定义网络3.定义损失函数和优化器4.训练网络并更新网络参数
- 最近需要用Python写一个简易通讯录,但是对于数据存储很发愁。大家都知道,使用 Python 中的列表和字典进行存储数据是很不靠谱的,所以
- 内容摘要:在本人上一篇教程《彻底弄懂CSS盒子模式五(定位强化练习) 》有讲到一个很酷的链接面板提示的实例制作,那时主要是用到di
- 一、利用ASP和ADO实现数据库操作的工作流程ASP内嵌了五个对象Resquest、Response、Server、Seesion、Appl
- Python3标准库操作系统接口os模块提供了不少与操作系统相关联的函数。>>> import os>>>
- 文中用到了BeautifulSoup这个库, 目的是处理html文档分析的, 因为我只是提取了title的关键字,所以可以用正则表达式代替,
- 最近在做cocos2d-x的简明配置,发现有的朋友的文本编辑器,自动将\r\n截断成\n,(在unix上换行使用\n,windows上,换行
- jQuery中文入门指南,翻译加实例,jQuery的起点教程中文版译者:Keel此文以实例为基础一步步说明了jQuery的工作方式。现以中文
- 在用Pycharm运行项目时,不知道干了什么,导致运行后无法像平时那样显示输出结果:最后试了很多种方法,终于解决了:将打勾的地方去掉,不要运
- 不用整天为美化select控件烦恼了。1、可批量美化select控件。2、可以有onchange句柄。3、触发onchange的函数可带参数
- 姓名的翻译: 英语是名(First name)在前,姓(Last name)在后。中文地址的翻译:如果你英语水平不高,填表时只要国家名用英语
- 测试靶机为DVWA,适合DVWA暴力破解模块的Low和Medium等级关键代码解释url指定url地址url = &qu
- Javascript 选择器(selector engine)似乎从 jQuery 流行以来就大行其道,改变了原有 Javascript 选
- lstm(*input, **kwargs)将多层长短时记忆(LSTM)神经网络应用于输入序列。参数:input_size:输入'x
- 好了,看看我们的代码吧:upload.htm' 上传页面<html> <body>&nb
- 本文实例讲述了python实现基于两张图片生成圆角图标效果的方法。分享给大家供大家参考。具体分析如下:使用pil的蒙版功能,将原图片和圆角图
- 这个javascript农历日历,万年历代码网上看到的,很不错,功能齐全,值得收藏!功能介绍:动态显示当前世界各国各时区时间,显示当前农历,
- OpenCV imread读取图片失败问题使用一下代码读取一张图片失败(不管是绝对路径还是相对路径,都失败),工程运行都没问题,就是图片读取
- 目前整个开发社区对AOP(Aspect Oriented Programing)推崇备至,也涌现出大量支持AOP的优秀Framework,-