ffmpy3与ffmpeg的简单使用

安装

python ffmpy3与ffmpeg的安装

https://blog.csdn.net/qq_40962368/article/details/90748852

ffmpy3的介绍:

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

        ffmpy3是一个用于FFmpeg的Python包装器,最初是从ffmpy项目派生出来的。它根据提供的参数及其各自的选项编译FFmpeg命令行,并使用Python的子进程执行它。

        ffmpy3类似于FFmpeg使用的命令行方法。它可以读取任意数量的输入“文件”(常规文件、管道、网络流、抓取设备等),并写入任意数量的输出“文件”。有关FFmpeg命令行选项和参数如何工作的详细信息,请参阅FFmpeg文档。

        ffmpy3支持FFmpeg的管道协议。这意味着可以将输入数据传递到stdin并从stdout获得输出数据。

        目前,ffmpy3已经为ffmpeg和ffprobe命令提供了包装器,但是应该可以用它运行其他ffmpeg工具(例如ffserver)。

FFmpeg的简单使用

一、格式转换

        最简单的用法例子是将媒体从一种格式转换为另一种格式(在本例中是从MP4转换为MPEG传输流),同时保留所有其他属性:

from ffmpy3 import FFmpeg
ff = FFmpeg(inputs={'test.mp4': None},
            outputs={'output.ts': None})
print(ff.cmd)
ff.run()

 

 

 

二、转码

        如果同时我们想用不同的解码器重新编码视频和音频,我们必须指定额外的输出选项:

        下面的例子:将音频编码为mp2格式,将视频编码为mpeg2video模式

from ffmpy3 import FFmpeg
# 这里a就是指audio,v就是指video
ff = FFmpeg(inputs={'test.mp4': None},
            outputs={'output.ts': '-c:a mp2 -c:v mpeg2video'})
print(ff.cmd)
ff.run()
ffmpeg -i test.mp4 -c:a mp2 -c:v mpeg2video output.ts



Input #0, ogg, from 'test.mp4':

Stream #0:1: Video: theora, yuv420p, 320x176, 25 fps, 25 tbr, 25 tbn, 25 tbc

Stream #0:2: Audio: vorbis, 48000 Hz, stereo, fltp, 160 kb/s



Stream mapping:

Stream #0:1 -> #0:0 (theora (native) -> mpeg2video (native))

Stream #0:2 -> #0:1 (vorbis (native) -> mp2 (native))



Output #0, mpegts, to 'output.ts':

Stream #0:0: Video: mpeg2video (Main), yuv420p, 320x176, q=2-31, 200 kb/s, 25 fps, 90k tbn, 25 tbc

Stream #0:1: Audio: mp2, 48000 Hz, stereo, s16, 384 kb/s

三、多路分解

        一个更复杂的使用示例是将MPEG传输流解复用为单独的基本(音频和视频)流,并将它们保存在保存编解码器的MP4容器中(注意这里如何使用列表作为选项):

        实现的效果就是音频与视频分离,音频放入了audio.mp4文件,视频在video.mp4文件

from ffmpy3 import FFmpeg
ff = FFmpeg(inputs={'input.ts': None},
            outputs={'video.mp4': ['-map', '0:0', '-c:a', 'copy', '-f', 'mp4'],
                     'audio.mp4': ['-map', '0:1', '-c:a', 'copy', '-f', 'mp4']})
print(ff.cmd)
ff.run()

ffmpeg -i input.ts -map 0:0 -c:a copy -f mp4 video.mp4 -map 0:1 -c:a copy -f mp4 audio.mp4



Output #1, mp4, to 'audio.mp4':

Stream #1:0: Audio: mp2 (mp4a / 0x6134706D), 48000 Hz, stereo, fltp, 384 kb/s



Output #0, mp4, to 'video.mp4':

Stream #0:0: Video: h264 (libx264) (avc1 / 0x31637661), yuv420p, 320x176 [SAR 1:1 DAR 20:11], q=-1--1, 25 fps, 12800 tbn, 25 tbc

        注意:不能混合选项的表达式格式,也就是说,不能有包含空格字符串的列表,除非不含有列表,将命令行作为一个整体字符串,即:['-map', '0:0', '-c:a', 'copy', '-f', 'mp4'] 或 '-map 0:0 -c:a copy -f mp4'

四、多路复用

        将多路视频和音频重新编码回MPEG传输流,即音频与视频的合成:

from ffmpy3 import FFmpeg
ff = FFmpeg(inputs={'video.mp4': None, 'audio.mp4': None},
            outputs={'output.ts': '-c:v h264 -c:a ac3'})
print(ff.cmd)
ff.run()
Stream mapping:

Stream #0:0 -> #0:0 (h264 (native) -> h264 (libx264))

Stream #1:0 -> #0:1 (mp3 (mp3float) -> ac3 (native))

        在某些情况下,必须保留输入和输出的顺序(例如使用FFmpeg -map选项时)。在这些情况下,使用常规Python字典将不起作用,因为它不能保持顺序。相反,使用OrderedDict。例如,我们希望将一个视频和两个音频流多路复用到一个MPEG传输流中,使用不同的编解码器对两个音频流进行重新编码。这里,我们使用OrderedDict来保存输入的顺序,以便它们匹配输出选项中的流的顺序:

from ffmpy3 import FFmpeg
from collections import OrderedDict
inputs = OrderedDict([('video.mp4', None), ('audio_1.mp3', None), ('audio_2.mp3', None)])
outputs = {'output.ts': '-map 0 -c:v h264 -map 1 -c:a:0 ac3 -map 2 -c:a:1 mp2'}
ff = FFmpeg(inputs=inputs, outputs=outputs)
print(ff.cmd)
ff.run()

        简化输出后的结果为: 

ffmpeg -i video.mp4 -i audio_1.mp3 -i audio_2.mp3 -map 0 -c:v h264 -map 1 -c:a:0 ac3 -map 2 -c:a:1 mp2 output.ts



Input #0, mov,mp4,m4a,3gp,3g2,mj2, from 'video.mp4':

Stream #0:0(und): Video: h264 (High) (avc1 / 0x31637661), yuv420p, 320x176 [SAR 1:1 DAR 20:11], 241 kb/s, 25 fps, 25 tbr, 12800 tbn, 50 tbc (default)



Input #1, mov,mp4,m4a,3gp,3g2,mj2, from 'audio_1.mp3':

Stream #1:0(und): Audio: mp3 (mp4a / 0x6134706D), 48000 Hz, stereo, fltp, 384 kb/s (default)



Input #2, mov,mp4,m4a,3gp,3g2,mj2, from 'audio_2.mp3':

Stream #2:0(und): Audio: mp3 (mp4a / 0x6134706D), 48000 Hz, stereo, fltp, 384 kb/s (default)



Stream mapping:

Stream #0:0 -> #0:0 (h264 (native) -> h264 (libx264))

Stream #1:0 -> #0:1 (mp3 (mp3float) -> ac3 (native))

Stream #2:0 -> #0:2 (mp3 (mp3float) -> mp2 (native))



Output #0, mpegts, to 'output.ts':

Stream #0:0(und): Video: h264 (libx264), yuv420p(progressive), 320x176 [SAR 1:1 DAR 20:11], q=-1--1, 25 fps, 90k tbn, 25 tbc (default)

Stream #0:1(und): Audio: ac3, 48000 Hz, stereo, fltp, 192 kb/s (default)

Stream #0:2(und): Audio: mp2, 48000 Hz, stereo, s16, 384 kb/s (default)

五、使用管道协议

        ffmpy3可以从STDIN(standard input)读取输入,并将输出写入STDOUT(standard output)。这可以通过使用FFmpeg管道协议来实现。下面的示例从包含RGB格式原始视频帧的文件中读取数据,并通过STDIN将其传递给ffmpy3;ffmpy3将用H.264编码原始帧数据,并将其打包到一个MP4容器中,将输出传递给STDOUT(注意,必须使用子进程将STDOUT重定向到管道中。管道为stdout值,否则输出将丢失):

from ffmpy3 import FFmpeg
import subprocess
ff = FFmpeg(inputs={'pipe:0': '-f rawvideo -pix_fmt rgb24 -s:v 640x480'},
            outputs={'pipe:1': '-c:v h264 -f mp4'})
print(ff.cmd)
stdout, stderr = ff.run(input_data=open('rawvideo', 'rb').read(), stdout=subprocess.PIPE)

六、异步执行

        在某些情况下,您可能不希望运行FFmpeg并阻塞等待结果,或者在应用程序中引入多线程。在这种情况下,使用asyncio进行异步执行是可能的。

        也可以在不使用多线程或阻塞的情况下处理FFmpeg输出。

七、复杂命令行

        FFmpeg命令行选项可能非常复杂,比如在使用过滤时。因此,理解使用ffmpy3构建命令行的一些规则非常重要。如果一个选项包含引号,则必须在没有引号的选项列表中作为单独的项指定。但是,如果选项使用一个字符串,则必须将被引用的选项的引号保存在字符串中:

from ffmpy3 import FFmpeg
ff = FFmpeg(inputs={'input.ts': None},
            outputs={'output.ts': ['-vf', 'adif=0:-1:0, scale=iw/2:-1']})
print(ff.cmd)
ff = FFmpeg(inputs={'input.ts': None},
            outputs={'output.ts': '-vf "adif=0:-1:0, scale=iw/2:-1"'})
print(ff.cmd)
ffmpeg -i input.ts -vf "adif=0:-1:0, scale=iw/2:-1" output.ts
ffmpeg -i input.ts -vf "adif=0:-1:0, scale=iw/2:-1" output.ts

Process finished with exit code 0

        一个更复杂的例子是命令行,它将时间码烧录成视频:

ffmpeg -i input.ts -vf "drawtext=fontfile=/Library/Fonts/Verdana.ttf: timecode='09\:57\:00\:00': r=25: x=(w-tw)/2: y=h-(2*lh): fontcolor=white: box=1: boxcolor=0x00000000@1" -an output.ts

        在ffmpy3中可以表示为:

from ffmpy3 import FFmpeg
ff = FFmpeg(inputs={'input.ts': None},
            outputs={'output.ts': ['-vf', "drawtext=fontfile=/Library/Fonts/Verdana.ttf: timecode='09\:57\:00\:00': r=25: x=(w-tw)/2: y=h-(2*lh): fontcolor=white: box=1: boxcolor=0x00000000@1", '-an']})
print(ff.cmd)

        相同的命令行可以编译通过传递输出选项作为一个字符串,同时保留引号:

from ffmpy3 import FFmpeg
ff = FFmpeg(inputs={'input.ts': None},
            outputs={'output.ts': "-vf \"drawtext=fontfile=/Library/Fonts/Verdana.ttf: timecode='09\:57\:00\:00': r=25: x=(w-tw)/2: y=h-(2*lh): fontcolor=white: box=1: boxcolor=0x00000000@1\" -an"})
print(ff.cmd)

  • 8
    点赞
  • 62
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值