0%

FFmpeg常用命令

下载已编译的静态库:https://evermeet.cx/ffmpeg/

命令分类

  • 基本信息查询
  • 录制
  • 分解、复用
  • 处理原始数据
  • 裁剪与合并
  • 图片/视频互转
  • 直播
  • 滤镜

基本信息查询

版本信息:

  • -version

支持的分解、复用:

  • -demuxers
  • -muxers

支持的设备:

  • -devices

支持的(libavcodec已知的)编解码器:

  • -codecs
  • -decoders
  • -encoders

libavfilter支持的码流滤镜:

  • -bsfs

支持的格式:

  • -formats,支持的文件格式
  • -protocols,网络协议
  • -pix_fmts,像素格式
  • -sample_fmts,采样格式
  • -layouts,声道布局

支持的滤镜:

  • -filters

支持的颜色名称:

  • -colors

命令基本格式与参数

ffmpeg命令基本格式:

1
2
ffmpeg [全局参数] {[输入文件参数] -i 输入文件URL} ...\
{[输出文件参数] 输出文件URL} ...

默认情况下,FFmpeg只包含输入文件中每种媒体类型(音频、视频、字幕)的一个流,并将其添加到每个输出文件中。它根据一下规则选择每种媒体类型中的流:

  • 视频,选择最高分辨率的流。
  • 音频,选择声道最多的流。
  • 字幕,选择第一个流。

当然上述规则相等,则优先选取最低索引值的流。

可以通过-vn-an-sn-dn来禁用某些媒体类型。要进行全面的手动控制,则使用-map选项。

ffmpeg通过-i选项读取任意数量的输入(可以是文件、管道、网络流、抓取设备等),并写入任意数量的输出。原则上,每个输入/输出都可以包含任意数量的不同类型的媒体流(视频、音频、字幕、附件/数据)。流的数量和/或类型是由容器格式来限制,选择从哪个输入进来到哪个输出将自动完成,如需控制则使用-map选项。

要引用选项中的输入,必须使用索引(从0开始)。例如,第一个输入是0。文件中的媒体流也可以通过索引引用,如2:3是指第三个输入中的第四个流。

主要参数

  • -f fmt,输入/输出强制的文件格式。格式通常可以通过扩展名中猜测出来,因此不常使用。
  • -i url,输入URL。
  • -y,全局参数,覆盖输出文件而不询问。
  • -n,全局参数,不覆盖输出文件,如果存在指定文件则退出。
  • -c [:stream_specifier] codec-codec [:stream_specifier] codec,输入、输出、单个流,选择一个解码器(在输入文件之前使用)或编码器(在输出文件前使用),可用于一个或多个流。传递编码器名称或copy(仅输出)表示不重新编码。
  • -t duration,输入、输出,当用作输入选项(在-i之前),表示限制从输入文件读取的数据的时长;当用作输出选项时(在输出url之前),表示在到达时长之后停止输出。
  • -ss position,输入、输出,当用作输入选项(在-i之前),表示在输入文件中寻找位置。注意,在大多数格式中,不可能精确搜索,因此ffmpeg将在位置之前寻找最近的点。当转码和-accureate_seek被启用时(默认),搜索点和指定的位置之间的额外分段将被解码和丢弃。当进行流式复制或使用-noaccureate_seek时,它将被保留。当用作输出选项(在输出url之前),解码但丢弃输入,知道时间戳到达指定的位置。
  • frames [:stream_specifier] framecount,输出、单个流,停止在给定帧数量后写入流。
  • filter [:stream_specifier] filtergraph,输出、单个流,创建由filtergraph指定的滤镜链图,并将其进行处理流。filtergraph的流必须具有相同类型的单个输入和单个输出。在filtergraph中,输入与标签相关联,标签与输出相关联。

视频参数

  • -vframes num,输出,设置要输出的视频帧数量。
  • -r [:stream_specifier] fps,输入、输出、单个流,设置帧率(单位Hz)。
    • 作为输入选项,忽略存储文件中的让和时间戳,根据速率生成新的时间戳。这与用于-framerate选项不同(旧版是相同的)。
    • 作为输出选项,复制或丢弃输入帧以实现很定输出帧率。
  • -s [:stream_specifier] size,输入、输出、单个流,设置窗口大小。
    • 作为输入选项,与video_size相同,由某些分帧器识别,其帧尺寸未存储在文件中。
    • 作为输出选项,这将会把缩放视频滤镜传输到相应的滤镜链图末尾。请直接使用比例滤镜插入滤镜链图开头或其他地方。格式是宽x高
  • -aspect [:stream_specifier] ratio,输出、单个流,指定视频的宽高比。值可以是浮点数字也可以是w:h字符串,如4:316:9。如果与-vcodec副本一起使用,会影响存储在容器级别的宽高比,但不会影响存储在编码帧中的宽高比(如果存在的话)。
  • -vn,输出,禁用视频轨道。
  • -vf filtergraph,输出,创建由filtergraph指定的滤镜链图,并使用它来处理流。

音频参数

  • -aframes num,输出,设置输出音频的帧数。
  • -ar [:stream_specifier] freq,输入、输出、单个流,设置音频采样率。对于输入流,默认设置为输入流的采样率。对于输出流,该选项仅适用于音频设备采集和原始分路器,并映射到相应的分路器选件。
  • -an,输出,禁用音频轨道。
  • -acodec name,输入、输出,设置音频编解码器。
  • -sample_fmt [:stream_specifier] fmt,输出、单个流,设置音频采样格式。使用-sample_fmts可以获得支持的采样格式列表。
  • -af filtergraph,输出,创建由filtergraph指定的滤镜链图,并用它来处理音频。

录制

录制屏幕:

1
2
3
4
5
6
7
8
9
10
# 录制纯视频
ffmpeg -f avfoundation -i 1 -r 30 out.yuv
# 播放纯视频
ffplay -s 3360x2100 -pix_fmt uyvy422 out.yuv

# 录制音视频
ffmpeg -f avfoundation -i 1:0 -r 29.97
-c:v libx264 -crf 0
-c:a libfdk_aac -profile:a aac_he_v2 -b:a 32k
out.flv
  • -f,指定使用avfoundation进行采集。
  • -i,指定输入索引。
  • -r,指定帧率。
  • -crf,x264参数,0表示无损压缩。
  • -b:a,指定音频码率。

录制后,会输出录制的格式,后面进行播放要传入这些信息才能正确播放。

列出支持的设备:

1
2
3
4
5
6
7
ffmpeg -f avfoundation -list_devices true -i ""
[AVFoundation indev @ 0x7fa856d0c600] AVFoundation video devices:
[AVFoundation indev @ 0x7fa856d0c600] [0] FaceTime高清摄像头(内建)
[AVFoundation indev @ 0x7fa856d0c600] [1] Capture screen 0
[AVFoundation indev @ 0x7fa856d0c600] [2] Capture screen 1
[AVFoundation indev @ 0x7fa856d0c600] AVFoundation audio devices:
[AVFoundation indev @ 0x7fa856d0c600] [0] Built-in Microphone

摄像头录制:

1
ffmpeg -framerate 30 -f avfoundation -i 0:0 out.mp4

录制音频:

1
2
3
4
# 录制
ffmpeg -f avfoundation -i :0 output.wav
# 播放,由于存成wav,所以已经带上了音频的格式,直接播放即可。
ffplay output.wav

分解与复用

对容器内的数据重新组装,当然这样做的前提是对容器媒体文件进行分解形成编码数据包,完成重新组装后还需要进行重新封装。整个过程不对编码数据包做修改,只是换了个马甲。

1
2
3
4
5
6
7
8
9
# 换媒体容器
ffmpeg -i local.mp4 -vcodec copy -acodec copy local_output.mov
ffmpeg -i local.mp4 -vcodec copy -acodec copy local_output.mkv

# 抽取视频,当然h264包含了SPS、PPS,可以直接播放
ffmpeg -i local.mp4 -vcodec copy -an local_output.h264

# 抽取音频
ffmpeg -i input.mp4 -acodec copy -vn out.aac

这里面的关键是codec copy,即忽略指定流的编解码步骤,但同时功能也会受限(例如不能使用滤镜,因为滤镜是作用于未压缩的数据),只能进行多路分解和多路复用。对更改容器格式或修改容器级元数据很有用。

整个过程非常快,且没有质量损失。

处理原始数据

这里的原始数据是指解码后的数据。

1
2
3
4
5
6
7
8
9
# 提取YUV数据,关键是使用 -c:v rawvideo
ffmpeg -i input.mp4 -an -c:v rawvideo -pix_fmt yuv420p out.yuv
# 播放YUV
ffplay -s 1280x720 -pix_fmt yuv420p out.yuv

# 提取PCM数据
ffmpeg -i local.mp4 -vn -ar 44100 -ac 2 -f s16le out.pcm
# 播放,PCM是不带元数据的,所以播放时需要执行音频格式
ffplay -ar 44100 -ac 2 -f s16le out.pcm
  • -c:v,指定的是使用视频编码器,把它换成-vcodec也是同样的效果。
  • -pix_fmt,指定像素格式。
  • -ar,音频采样率。
  • -ac,声道数。
  • -f,数据存储格式。
    • s16le表示sign的16位小端模式整数。

注意这里的-vcodec一定要选择rawvideo,而不是copy;类似的,音频则直接不设置-acodec,而是直接接音频的参数。否则它们输出的结果都不是原始数据。

滤镜

使用简单的视频画幅裁剪滤镜:

1
2
3
ffmpeg -i input.mov
-vf crop=in_w-200:in_h-200
-c:v libx264 -c:a copy out.mp4
  • -vf,视频简单滤镜。
    • crop滤镜名称,后面等号接滤镜参数,in_win_h引用了原视频的宽高,并使用冒号拼接参数。

裁剪与合并

1
2
3
4
5
6
7
# 裁剪
ffmpeg -i input.mp4
-ss 00:00:00 -t 10
out.ts

# 合并,文本内容每一行必须为 `file '文件路径'`
ffmpeg -f concat -i inputs.txt out.flv

图片/视频互转

1
2
3
4
5
# 视频 -> 图片
ffmpeg -i input.flv -r 1 -f image2 output-%3d.jpeg

# 图片 -> 视频
ffmpeg -i input-%3d.jpg out.mp4
  • -r,fps,每秒转一张图片。
  • -f,转换的格式。

直播推流/拉流

1
2
3
4
5
# 直播推流
ffmpeg -re -i input.mp4 -c copy -f flv rtmp://server/live/stream_name

# 拉流
ffmpeg -i rtmp://server/live/stream_name -c copy dump.flv
  • -re,减慢帧率使其与播放时的帧率保持同步。
  • -c copy,音视频不转码。
  • -f,指定推流的文件格式。

推流的格式一定需要与拉取的格式对应上。

欢迎关注我的其它发布渠道