使用 ffmpeg 获取帧数
Posted
技术标签:
【中文标题】使用 ffmpeg 获取帧数【英文标题】:Fetch frame count with ffmpeg 【发布时间】:2011-01-02 08:03:27 【问题描述】:有谁知道如何使用 ffmpeg 从视频文件中获取总帧数? ffmpeg 的渲染输出显示当前帧,我需要帧数来计算进度百分比。
【问题讨论】:
superuser.com/questions/84631/… 【参考方案1】:ffprobe
ffprobe -v error -select_streams v:0 -count_packets -show_entries stream=nb_read_packets -of csv=p=0 input.mp4
这实际上是对数据包而不是帧进行计数,但速度要快得多。结果应该是一样的。如果您想通过计数帧来验证,请将 -count_packets
更改为 -count_frames
和 nb_read_packets
更改为 nb_read_frames
。
ffprobe 选项的含义
-v error
这隐藏了“信息”输出(版本信息等),这使得解析更容易(但如果您寻求帮助,则会更难,因为它隐藏了重要信息)。
-count_frames
统计每个流的包数,并在对应的流部分上报。
-select_streams v:0
只选择第一个视频流。
-show_entries stream=nb_read_packets
仅显示nb_read_frames
的条目。
-of csv=p=0
设置输出格式。在这种情况下,它会隐藏描述并仅显示值。有关其他格式(包括 JSON)的信息,请参阅 FFprobe Writers。
只计算关键帧
见Checking keyframe interval?
MP4 编辑列表
MP4/M4V/M4A/MOV 中存在的编辑列表会影响您的帧数。
另见
FFprobe Documentation FFmpeg Wiki: FFprobe Tips媒体信息
众所周知的mediainfo
工具可以输出帧数:
mediainfo --Output="Video;%FrameCount%" input.avi
MP4 盒子
适用于 MP4/M4V/M4A 文件。
来自gpac的MP4Box
可以显示帧数:
MP4Box -info input.mp4
请参阅输出中的Media Info
行以获取相关视频流:
Media Info: Language "Undetermined (und)" - Type "vide:avc1" - 2525 samples
在本例中,视频流有 2525 帧。
boxdumper
适用于 MP4/M4V/M4A/MOV 文件。
boxdumper
是来自 l-smash 的一个简单工具。它会输出大量信息。在stsz
样本大小框部分下,请参阅sample_count
了解帧数。在这个例子中,输入有 1900 个视频帧:
boxdumper input.mp4
...
[stsz: Sample Size Box]
position = 342641
size = 7620
version = 0
flags = 0x000000
sample_size = 0 (variable)
sample_count = 1900
请注意,一个文件可能有多个 stsz
原子。
【讨论】:
或者,如果您想要更快的速度并且如果 nb_frames 足够可靠,请简化为:ffprobe -v error -select_streams v:0 -show_entries stream=nb_frames -of default=nokey=1:noprint_wrappers=1 input.mkv
这会为我输出两次答案(即 2600 \n 2600)。有什么特别的原因会发生吗?
@jbodily 我的例子还是 juanitogan 的?我不能使用任何一个复制它。在这里工作不多。
+1,尤其是因为,与任何命令行工具的许多其他答案不同,这个答案实际上解释了所有命令行选项。谢谢。
请注意,第一个选项,查询容器,由于 count_frames 而实际处理文件。请参阅@juanitogan 的评论。【参考方案2】:
在 Unix 中,这就像一个魅力:
ffmpeg -i 00000.avi -vcodec copy -acodec copy -f null /dev/null 2>&1 \
| grep 'frame=' | cut -f 2 -d ' '
【讨论】:
真的很不错。只是您不需要复制音频流。您可以使用 -an instat。 @PrakharMohanSrivastava 检查this answer 实际上,这似乎又快又可靠:ffmpeg -i 00000.avi -map 0:v:0 -c copy -f null -y /dev/null 2>&1 | grep -Eo 'frame= *[0-9]+ *' | grep -Eo '[0-9]+' | tail -1
@Michael 感谢我早上喝咖啡时的微笑 :-)
@TimothyZorn 你成就了我的一天!【参考方案3】:
改为根据时间计算。
这就是我所做的,它对我和其他许多人都很有效。首先,在下面的sn-p中找到视频的长度:
Seems stream 0 codec frame rate differs from container frame rate: 5994.00
(5994/1) -> 29.97 (30000/1001)
Input #0, mov,mp4,m4a,3gp,3g2,mj2, from '/Users/stu/Movies/District9.mov':
Duration: 00:02:32.20, start: 0.000000, bitrate: 9808 kb/s
Stream #0.0(eng): Video: h264, yuv420p, 1920x1056, 29.97tbr, 2997tbn, 5994tbc
Stream #0.1(eng): Audio: aac, 44100 Hz, 2 channels, s16
Stream #0.2(eng): Data: tmcd / 0x64636D74
您应该能够始终如一地安全地找到 Duration: hh:mm:ss.nn
以确定源视频剪辑的大小。然后,对于每个更新行(CR,无 LF),您可以解析当前时间标记的文本:
frame= 84 fps= 18 q=10.0 size= 5kB time=1.68 bitrate= 26.1kbits/s
frame= 90 fps= 17 q=10.0 size= 6kB time=1.92 bitrate= 23.8kbits/s
frame= 94 fps= 16 q=10.0 size= 232kB time=2.08 bitrate= 913.0kbits/s
请注意不要总是期望这些状态行有完美的输出。它们可以包含如下错误消息:
frame= 24 fps= 24 q=-1.0 size= 0kB time=1.42 bitrate= 0.3kbits/s
frame= 41 fps= 26 q=-1.0 size= 0kB time=2.41 bitrate= 0.2kbits/s
[h264 @ 0x1013000]Cannot parallelize deblocking type 1, decoding such frames in
sequential order
frame= 49 fps= 24 q=26.0 size= 4kB time=0.28 bitrate= 118.1kbits/s
frame= 56 fps= 22 q=23.0 size= 4kB time=0.56 bitrate= 62.9kbits/s
一旦你有时间,它就是简单的数学运算:time / duration * 100 = % done
。
【讨论】:
对不起,我很愚蠢,但是当持续时间为 hh:mm:ss.nn 格式且时间始终为 xx.yy 格式时,我该怎么做时间/持续时间? @Omar,作为 .NET 开发人员,我要做的是从中创建一个TimeSpan
,然后使用 currentDurationTimeSpan.Ticks / (totalDurationTimeSpan.Ticks / 100)
。 TimeSpan 还提供了强大的 Parse 功能,check it out
优秀的解决方案,我的时间是 hh:mm:ss:ms 所以我想在这 3 年里 FFMPEG 改进了输出时间格式。
请注意,控制台输出可能会显示 29.97,但它是 30000/1001 的缩写。 23.98 相同,即 24000/1001,59.94 为 60000/1001。
请注意,这不适用于可变帧率视频(显然)。【参考方案4】:
并非所有格式都存储它们的帧数或总持续时间 - 即使它们存储,文件也可能不完整 - 因此 ffmpeg 在默认情况下无法准确检测到它们中的任何一个。
相反,请尝试寻找文件末尾并读取时间,然后一边走一边计算当前时间。
或者,您可以尝试AVFormatContext->nb_index_entries
或检测到的持续时间,这应该可以在至少未损坏的 AVI/MOV 或库 FFMS2 上正常工作,这可能太慢而无法使用进度条。
【讨论】:
【参考方案5】:您可以使用ffprobe
通过以下命令获取帧号
-
第一种方法
ffprobe.exe -i video_name -print_format json -loglevel fatal -show_streams -count_frames -select_streams v
告诉以json
格式打印数据
select_streams v
会告诉ffprobe
只给我们video
流数据,如果你删除它,它也会给你audio
信息
输出会像
"streams": [
"index": 0,
"codec_name": "mpeg4",
"codec_long_name": "MPEG-4 part 2",
"profile": "Simple Profile",
"codec_type": "video",
"codec_time_base": "1/25",
"codec_tag_string": "mp4v",
"codec_tag": "0x7634706d",
"width": 640,
"height": 480,
"coded_width": 640,
"coded_height": 480,
"has_b_frames": 1,
"sample_aspect_ratio": "1:1",
"display_aspect_ratio": "4:3",
"pix_fmt": "yuv420p",
"level": 1,
"chroma_location": "left",
"refs": 1,
"quarter_sample": "0",
"divx_packed": "0",
"r_frame_rate": "10/1",
"avg_frame_rate": "10/1",
"time_base": "1/3000",
"start_pts": 0,
"start_time": "0:00:00.000000",
"duration_ts": 256500,
"duration": "0:01:25.500000",
"bit_rate": "261.816000 Kbit/s",
"nb_frames": "855",
"nb_read_frames": "855",
"disposition":
"default": 1,
"dub": 0,
"original": 0,
"comment": 0,
"lyrics": 0,
"karaoke": 0,
"forced": 0,
"hearing_impaired": 0,
"visual_impaired": 0,
"clean_effects": 0,
"attached_pic": 0
,
"tags":
"creation_time": "2005-10-17 22:54:33",
"language": "eng",
"handler_name": "Apple Video Media Handler",
"encoder": "3ivx D4 4.5.1"
]
2。 你可以使用
ffprobe -v error -show_format -show_streams video_name
这将为您提供流数据,如果您想要选择帧速率等信息,请使用以下命令
ffprobe -v error -select_streams v:0 -show_entries stream=avg_frame_rate -of default=noprint_wrappers=1:nokey=1 video_name
根据您的视频信息给出一个数字,问题是当您使用此方法时,您可能会得到一个N/A
作为输出。
欲了解更多信息,请查看此页面FFProbe Tips
【讨论】:
【参考方案6】:由于我的评论得到了一些赞成,我想我会把它作为答案:
ffmpeg -i 00000.avi -map 0:v:0 -c copy -f null -y /dev/null 2>&1 | grep -Eo 'frame= *[0-9]+ *' | grep -Eo '[0-9]+' | tail -1
这应该很快,因为没有执行编码。 ffmpeg
将尽快解复用文件并读取(解码)第一个视频流。第一个grep
命令将抓取显示框架的文本。第二个grep
命令将只从中获取数字。 tail
命令将只显示最后一行(最终帧数)。
【讨论】:
【参考方案7】:尝试类似:
ffmpeg -i "path to file" -f null /dev/null
它将帧号写入标准错误,因此您可以从中检索最后一帧。
【讨论】:
【参考方案8】:试试这个:
ffmpeg -i "path to file" -f null /dev/null 2>&1 | grep 'frame=' | cut -f 2 -d ' '
【讨论】:
不适用于*.ts
。输出为空行。【参考方案9】:
对于 necro 的回答很抱歉,但可能需要这个(因为我没有为最近的 ffmpeg 版本找到解决方案。
使用 ffmpeg 3.3.4 我发现可以找到以下内容:
ffprobe -i video.mp4 -show_streams -hide_banner | grep "nb_frames"
最后它会输出帧数。它适用于我的音频视频。不过,它给出了两次“nb_frames”行,但第一行是我测试的视频的实际帧数。
【讨论】:
谢谢@acidrums4。验证此方法适用于我今天构建的 github 的最新版本。 使用ffprobe -i video.mp4 -show_streams -hide_banner | grep "nb_frames" | head -n1 | cut -d"=" -f2
为我工作,它将输出减少到只是数字。【参考方案10】:
以 stu 的回答为基础。以下是我如何从手机中找到视频的帧速率。我运行了以下命令一段时间。在我不耐烦并点击^C之前,我让帧数达到约10,000:
$ ffmpeg -i 2013-07-07\ 12.00.59.mp4 -f null /dev/null 2>&1
...
Press [q] to stop, [?] for help
[null @ 0x7fcc80836000] Encoder did not produce proper pts, making some up.
frame= 7989 fps= 92 q=0.0 Lsize=N/A time=00:04:26.30 bitrate=N/A dup=10 drop=0
video:749kB audio:49828kB subtitle:0 global headers:0kB muxing overhead -100.000042%
Received signal 2: terminating.
$
然后,我从以“frame=”开头的那一行中获取了两条信息,帧数为 7989,时间为 00:04:26.30。您首先需要将时间转换为秒,然后将帧数除以秒以获得“每秒帧数”。 “每秒帧数”是您的帧速率。
$ bc -l
0*60*60 + 4*60 + 26.3
266.3
7989/(4*60+26.3)
30.00000000000000000000
$
我的视频帧率为 30 fps。
【讨论】:
【参考方案11】:我能够做到这一点的唯一准确信息如下:
ffprobe -i my_video.mp4 -show_frames 2>&1|grep -c '^\[FRAME'
为了确保这适用于视频:
ffprobe -i my_video.mp4 -show_frames 2>&1 | grep -c media_type=video
【讨论】:
我赞成您的回答,但这仅在视频不包含音频时才有效。如果它确实包含,这个可以工作:ffprobe -i my_video.mp4 -show_frames 2>&1 | grep -c media_type=video
【参考方案12】:
我使用 php_ffmpeg 然后我可以获得电影的所有时间和所有帧。如下
$input_file='/home/strone/workspace/play/CI/abc.rmvb';
$ffmpegObj = new ffmpeg_movie($input_file);
echo $ffmpegObj->getDuration();
echo $ffmpegObj->getFrameCount();
然后详细信息在页面上。
http://ffmpeg-php.sourceforge.net/doc/api/ffmpeg_movie.php
【讨论】:
【参考方案13】:Cmd ->
ffprobe.exe -v error -select_streams v:0 -show_entries stream=r_frame_rate,duration -of default=nw=1 "d:\movies\The.Matrix.1999.1080p.BrRip.x264.YIFY.dut.mp4"
Result ->
r_frame_rate=24000/1001
duration=8177.794625
Calculation ->
Frames=24000/1001*8177.794625=196071
Proof ->
ffmpeg -i "d:\movies\The.Matrix.1999.1080p.BrRip.x264.YIFY.dut.mp4" -f null /dev/null
ffmpeg version N-92938-g0aaaca25e0-ffmpeg-windows-pacman Copyright (c) 2000-2019 the FFmpeg developers
built with gcc 8.2.0 (GCC)
configuration: --pkg-config=pkg-config --pkg-config-flags=--static --extra-version=ffmpeg-windows-pacman --enable-version3 --disable-debug --disable-w32threads --arch=x86_64 --target-os=mingw32 --cross-prefix=/opt/sandbox/cross_compilers/mingw-w64-x86_64/bin/x86_64-w64-mingw32- --enable-libcaca --enable-gray --enable-libtesseract --enable-fontconfig --enable-gmp --enable-gnutls --enable-libass --enable-libbluray --enable-libbs2b --enable-libflite --enable-libfreetype --enable-libfribidi --enable-libgme --enable-libgsm --enable-libilbc --enable-libmodplug --enable-libmp3lame --enable-libopencore-amrnb --enable-libopencore-amrwb --enable-libopus --enable-libsnappy --enable-libsoxr --enable-libspeex --enable-libtheora --enable-libtwolame --enable-libvo-amrwbenc --enable-libvorbis --enable-libvpx --enable-libwebp --enable-libzimg --enable-libzvbi --enable-libmysofa --enable-libaom --enable-libopenjpeg --enable-libopenh264 --enable-liblensfun --enable-nvenc --enable-nvdec --extra-libs=-lm --extra-libs=-lpthread --extra-cflags=-DLIBTWOLAME_STATIC --extra-cflags=-DMODPLUG_STATIC --extra-cflags=-DCACA_STATIC --enable-amf --enable-libmfx --enable-gpl --enable-avisynth --enable-frei0r --enable-filter=frei0r --enable-librubberband --enable-libvidstab --enable-libx264 --enable-libx265 --enable-libxvid --enable-libxavs --enable-avresample --extra-cflags='-march=core2' --extra-cflags=-O2 --enable-static --disable-shared --prefix=/opt/sandbox/cross_compilers/mingw-w64-x86_64/x86_64-w64-mingw32 --enable-nonfree --enable-decklink --enable-libfdk-aac
libavutil 56. 25.100 / 56. 25.100
libavcodec 58. 43.100 / 58. 43.100
libavformat 58. 25.100 / 58. 25.100
libavdevice 58. 6.101 / 58. 6.101
libavfilter 7. 47.100 / 7. 47.100
libavresample 4. 0. 0 / 4. 0. 0
libswscale 5. 4.100 / 5. 4.100
libswresample 3. 4.100 / 3. 4.100
libpostproc 55. 4.100 / 55. 4.100
Input #0, mov,mp4,m4a,3gp,3g2,mj2, from 'd:\movies\The.Matrix.1999.1080p.BrRip.x264.YIFY.dut.mp4':
Metadata:
major_brand : isom
minor_version : 512
compatible_brands: isomiso2avc1mp41
encoder : Lavf58.25.100
Duration: 02:16:17.91, start: 0.000000, bitrate: 2497 kb/s
Stream #0:0(und): Video: h264 (High) (avc1 / 0x31637661), yuv420p, 1920x800 [SAR 1:1 DAR 12:5], 2397 kb/s, 23.98 fps, 23.98 tbr, 24k tbn, 47.95 tbc (default)
Metadata:
handler_name : VideoHandler
Stream #0:1(und): Audio: aac (LC) (mp4a / 0x6134706D), 44100 Hz, stereo, fltp, 93 kb/s (default)
Metadata:
handler_name : GPAC ISO Audio Handler
Stream mapping:
Stream #0:0 -> #0:0 (h264 (native) -> wrapped_avframe (native))
Stream #0:1 -> #0:1 (aac (native) -> pcm_s16le (native))
Press [q] to stop, [?] for help
Output #0, null, to '/dev/null':
Metadata:
major_brand : isom
minor_version : 512
compatible_brands: isomiso2avc1mp41
encoder : Lavf58.25.100
Stream #0:0(und): Video: wrapped_avframe, yuv420p, 1920x800 [SAR 1:1 DAR 12:5], q=2-31, 200 kb/s, 23.98 fps, 23.98 tbn, 23.98 tbc (default)
Metadata:
handler_name : VideoHandler
encoder : Lavc58.43.100 wrapped_avframe
Stream #0:1(und): Audio: pcm_s16le, 44100 Hz, stereo, s16, 1411 kb/s (default)
Metadata:
handler_name : GPAC ISO Audio Handler
encoder : Lavc58.43.100 pcm_s16le
frame=196071 fps=331 q=-0.0 Lsize=N/A time=02:16:17.90 bitrate=N/A speed=13.8x
video:102631kB audio:1408772kB subtitle:0kB other streams:0kB global headers:0kB muxing overhead: unknown
【讨论】:
帧率通常由两个参数计算得出。 r_frame_rate=24000/1001 (=23,97602397602397...) 由 ffmpeg 四舍五入为:23.98 持续时间 = 小时*3600+分钟*60+秒。剩余时间 = 8177,91 而持续时间参数 = 8177.794625 但帧数=24000/1001*8177.794625 =196071 给出确切的帧数。 (不开玩笑)。【参考方案14】:Linux
ffmpeg -i "/home/iorigins/Завантаження/123.mov" -f null /dev/null
红宝石
result = `ffmpeg -i #path -f null - 2>&1`
r = result.match("frame=([0-9]+)")
p r[1]
【讨论】:
【参考方案15】:ffprobe
和 ffmpeg
信息的问题
是不是帧中的实际长度有一些差异。
此脚本尝试提取最后一帧。 成功的帧号也可以在搅拌机内部工作。 超出该数量的帧也无法在搅拌机中提取。
#!/usr/bin/env bash
# find the number of frames in a movie clip
FMAYBE=$(ffprobe -v error -select_streams v:0 -show_entries stream=nb_frames -of default=nokey=1:noprint_wrappers=1 $1)
let FMAYBE=$FMAYBE+1
FEMPTY="-"
while [ -n "$FEMPTY" ] ; do
let FMAYBE=$FMAYBE-1
echo "Trying $FMAYBE"
FEMPTY=$(ffmpeg -i $1 -vf select="between(n\,$FMAYBE\,$FMAYBE)" -vsync 0 /tmp/fmaybe%d.png 2>&1 | grep 'empty')
done
echo "Succeeds: $FMAYBE"
【讨论】:
以上是关于使用 ffmpeg 获取帧数的主要内容,如果未能解决你的问题,请参考以下文章