来自实践中的教训,有时候纯ffmpeg命令并不好使,还得需要ffmpy包装器

举报
码农飞哥 发表于 2021/09/08 16:49:28 2021/09/08
【摘要】 ffmpeg实战

您好,我是码农飞哥,感谢您阅读本文,欢迎一键三连哦
本文是ffmpeg使用的第二篇文章,主要是关于ffmpy的使用。
干货满满,建议收藏,需要用到时常看看。 小伙伴们如有问题及需要,欢迎踊跃留言哦~ ~ ~。

前言

如果对ffmpeg命令的使用还不熟练的小伙伴可以看下第一篇文章实战详细讲解ffmpeg命令的使用(来自一线的经验,视频合并&avi转MP4&补空白音频【收藏下来一定用的到】)
为什么要介绍ffmpy呢?这是因为在项目开发中将ffmpeg的命令集成到flask框架时碰到部分命令在Linux下不能执行的情况。当时很是郁闷!!!单独执行ffmpeg命令没有问题。但是集成到项目中就是不行。那叫一个郁闷呀。ffmpy3是一个用于FFmpeg的Python包装器,最初是从ffmpy项目派生出来的。

ffmpy库的引入

要想使用ffmpy。首先需要引入ffmpy库,我这里使用的版本是0.2.2。引入的命令是:

pip install ffmpy==0.2.2 -i https://pypi.douban.com/simple

这里使用了豆瓣云,因为原生的进行下载依赖库的速度比较慢。

ffmpy官方文档的地址

https://ffmpy3.readthedocs.io/en/latest/ffmpy3.html

ffmpy的介绍

ffmpy中的FFmpeg类有两个核心的方法。一个是类的构造方法,一个是run方法。

FFmpeg类的构造方法

FFmpeg(executable='ffmpeg', global_options=None, inputs=None, outputs=None)

FFmpeg类的构造方法,说白了就是生成一个FFmpeg类的实例。其各个参数的含义是:

  1. executable指的是查找ffmpeg的命令,默认情况下从环境变量PATH中取ffmpeg命令,如果没有配置环境变量的话,则覆盖execuable,传入ffmpeg程序的绝对路径。
  2. global_options:指的是ffmpeg命令的全局命令参数,比如:-y,-v等等
  3. inputs(dict): 传入的是一个字典,key为输入文件的地址,value为对输入文件的操作命令参数。默认情况下是带有 -i 命令参数的。
  4. outputs (dict): 传入的是一个字典,key为输出文件的地址,value为对输出文件的操作命令参数。

FFmpeg的run方法

FFmpeg的run方法的作用是执行FFmpeg命令。该方法的定义是:run(input_data=None, stdout=None, stderr=None)。各个参数的含义是:

  1. input_data: FFmpeg以字节形式处理(音频、视频等)等输入数据
  2. stdout: 将FFmpeg重定向stdout到的位置。默认为None,意味着没有重定向
  3. stderr: 将 FFmpeg 重定向stderr到的位置。默认为None,意味着没有重定向。

其他的方法,比如cmd方法是打印执行的ffmpeg命令。

ffmpy的使用

下面以webm格式的视频转码成mp4的视频为例。
原始命令是:ffmpeg -y -i D:\\ffmpeg_test\\1.webm -r 30 D:\\ffmpeg_test\\1.mp4

1. webm转mp4直接执行ffmpeg的写法是:

import os
def webm2mp4():
    command = "ffmpeg -y -i D:\\ffmpeg_test\\1.webm -r 30 D:\\ffmpeg_test\\1.mp4"
    os.system(command)

2. webm转mp4通过ffmpy执行的写法是:

from ffmpy import FFmpeg
def webm_to_mp4():
    ff = FFmpeg(inputs={'D:\\ffmpeg_test\\1.webm': '-y'},
                outputs={'D:\\ffmpeg_test\\1.mp4': ' -r 30'})
    ff.run()

3.多个视频合并

# ffmpeg -f concat -safe 0 -y -i D:\\ffmpeg_test\\filelist.txt -c copy -strict -2 D:\\ffmpeg_test\\concated.mp4
ff = FFmpeg(
            inputs={'D:\\ffmpeg_test\\filelist.txt': ' -f concat -safe 0 -y '},
            outputs={'D:\\ffmpeg_test\\concated.mp4': ' -c copy -strict -2 '})
ff.run()

4. 视频剪切

# ffmpeg -ss 15 -t 25 -i D:\\ffmpeg_test\\concated.mp4  -c:v libx264 -c:a aac -strict experimental D:\\ffmpeg_test\\clip.mp4
ff = FFmpeg(inputs={'D:\\ffmpeg_test\\concated.mp4': '-ss 15 -t 25'},
            outputs={'D:\\ffmpeg_test\\clip.mp4': '-c:v libx264 -c:a aac -strict experimental'})
ff.run()

在实际开发中远远没有这么简单,视频剪切的开始时间和结束时间都是可变的。另外就是输入文件和输出文件的地址也是可变的。那么在这种情况下,代码该如何写呢?这里主要涉及到一个入参的问题。下面做做一个演示:

  1. 第一种方式,将开始时间和时长放在global_options参数中
start = 15
end = 25
video_input_path = 'D:\\ffmpeg_test\\concated.mp4'
video_output_path = 'D:\\ffmpeg_test\\clip.mp4'
ff = FFmpeg(global_options={" -ss {0} -t {1}".format(start, end)},
            inputs={video_input_path: '-y'},
            outputs={video_output_path: " -c:v libx264 -c:a aac -preset ultrafast -strict experimental "})
ff.run()
  1. 第二种方式,将开始时间和时长作为输入操作,进行参数的传入。
start = 15
end = 25
video_input_path = 'D:\\ffmpeg_test\\concated.mp4'
video_output_path = 'D:\\ffmpeg_test\\clip.mp4'
ff = FFmpeg(inputs={video_input_path: " -ss {0} -t {1}".format(start, end)},
            outputs={video_output_path: " -c:v libx264 -c:a aac -preset ultrafast -strict experimental "})
ff.run()

ffmpeg默认情况下是单个线程执行的,也就是说一个命令只有一个线程执行。如果要提高性能的话可以使用多线程执行。其命令参数是 -thread。比如改成5个线程执行,则命令是:

 ff = FFmpeg(executable=ffmpeg.FFMPEG_PATH,
                global_options={" -ss {0} -t {1}".format(start, end)},
                inputs={video_input_path: '-y'},
                outputs={video_output_path: " -c:v libx264 -c:a aac -threads 5 -preset ultrafast -strict experimental "})
 ff.run()

4.音频同轨

音频同轨涉及到两个输入文件,一个输出文件。所以inputs参数就有两个键值对。

# ffmpeg -y -i D:\\ffmpeg_test\\org_video_sound_input.wav -i D:\\ffmpeg_test\\org_voice_input.wav  -filter_complex amix=inputs=2:duration=longest D:\\ffmpeg_test\\org_voice_output.wav

sound1_path = 'D:\\ffmpeg_test\\org_video_sound_input.wav'
sound2_path = 'D:\\ffmpeg_test\\org_voice_input.wav'
sound_concated_path = 'D:\\ffmpeg_test\\org_voice_output.wav'
ff = FFmpeg(inputs={sound1_path: '-y', sound2_path: None},
            outputs={sound_concated_path:
                         ' -filter_complex amix=inputs=2:duration=longest '})
ff.run()

总结

本文详细介绍了ffmpy的使用,希望对读者朋友们有所帮助。

我是码农飞哥,再次感谢您读完本文

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gLjACjLb-1629712012256)(./attachments/1629699514774.drawio.html)]

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。