最快的方法提取帧使用ffmpeg?

嗨,我需要从视频中提取帧使用ffmpeg..还有比这更快的方法吗:

ffmpeg -i file.mpg -r 1/1 $filename%03d.jpg

?

379707 次浏览

如果JPEG编码步骤过于性能密集,您可以始终将未压缩的帧存储为BMP图像:

ffmpeg -i file.mpg -r 1/1 $filename%03d.bmp

这样做的另一个优点是,通过将代码转换为JPEG进行量化,不会导致更多的质量损失。(PNG也是无损的,但编码时间往往比JPEG长得多。)

遇到了这个问题,这里有一个快速的比较。比较以下两种从38m07秒长的视频中每分钟提取一帧的方法:

time ffmpeg -i input.mp4 -filter:v fps=fps=1/60 ffmpeg_%0d.bmp

1 m36.029s

这需要很长时间,因为ffmpeg解析整个视频文件以获得所需的帧。

time for i in {0..39} ; do ffmpeg -accurate_seek -ss `echo $i*60.0 | bc` -i input.mp4   -frames:v 1 period_down_$i.bmp ; done

0 m4.689s

这要快20倍。我们使用快速查找到所需的时间索引并提取一帧,然后为每个时间索引调用ffmpeg几次。注意-accurate_seek 是默认值 ,并确保在输入视频-i选项之前添加-ss

注意,最好使用-filter:v -fps=fps=...而不是-r作为后者可能不准确。尽管票上标明是固定的,我仍然遇到了一些问题,所以最好谨慎行事。

如果你确切地知道要提取哪些帧,例如1、200、400、600、800、1000,尝试使用:

select='eq(n\,1)+eq(n\,200)+eq(n\,400)+eq(n\,600)+eq(n\,800)+eq(n\,1000)' \
-vsync vfr -q:v 2

我使用这个管道Imagemagick的蒙太奇从任何视频得到10帧预览。显然,帧数需要使用ffprobe来计算

ffmpeg -i myVideo.mov -vf \
select='eq(n\,1)+eq(n\,200)+eq(n\,400)+eq(n\,600)+eq(n\,800)+eq(n\,1000)',scale=320:-1 \
-vsync vfr -q:v 2 -f image2pipe -vcodec ppm - \
| montage -tile x1 -geometry "1x1+0+0<" -quality 100 -frame 1 - output.png

小的解释:

  1. 在ffmpeg表达式中,+代表OR, *代表and
  2. \,只是转义了,字符
  3. 没有-vsync vfr -q:v 2,它似乎不工作,但我不知道为什么-有人吗?
在我的情况下,我需要帧至少每秒钟。我使用了上面的“寻求”方法,但不知道我是否可以并行完成任务。我在这里使用了先进先出的N个进程: https://unix.stackexchange.com/questions/103920/parallelize-a-bash-for-loop/216475#216475 < / p >
open_sem(){
mkfifo /tmp/pipe-$$
exec 3<>/tmp/pipe-$$
rm /tmp/pipe-$$
local i=$1
for((;i>0;i--)); do
printf %s 000 >&3
done
}
run_with_lock(){
local x
read -u 3 -n 3 x && ((0==x)) || exit $x
(
"$@"
printf '%.3d' $? >&3
)&
}
N=16
open_sem $N
time for i in {0..39} ; do run_with_lock ffmpeg -ss `echo $i` -i /tmp/input/GOPR1456.MP4  -frames:v 1 /tmp/output/period_down_$i.jpg  & done

从本质上讲,我用&但是将并发线程数限制为N。

在我的例子中,这将“寻求”方法从26秒提高到16秒。唯一的问题是主线程不能干净地退出到终端,因为stdout被淹没了。

我试过了。3600帧32秒。你的方法太慢了。你应该试试这个。

ffmpeg -i file.mpg -s 240x135 -vf fps=1 %d.jpg

这对我很有效

ffmpeg -i file.mp4 -vf fps=1 %d.jpg

每分钟输出一张图像,命名为img001.jpg, img002.jpg, img003.jpg等。%03d指示每个输出图像的序号将使用3位数字进行格式化。

ffmpeg -i myvideo.avi -vf fps=1/60 img%03d.jpg

fps=1/60更改为fps=1/30以每30秒捕获一个图像。类似地,如果你想每5秒捕捉一个图像,那么将fps=1/60更改为fps=1/5

来源:https://trac.ffmpeg.org/wiki/Create每X秒的视频缩略图

这比到目前为止所有其他命令都简单:

ffmpeg -i input.mp4 '%04d.png'

04更改为您需要容纳所有帧的任意数字。确保在数字之前总是有一个0,这样输出帧名是零填充的。

与@makeworld的答案相同,但也解决了这里关于帧数不一致和垂直同步的需要这里关于垂直同步的使用提到的问题:

// reliably works for jpg output (and probably png too)
ffmpeg -i rgb.mov -vf setpts=N/FR/TB -vsync 0 ./images/%05d.jpg


// reliably works for png output only
ffmpeg -i rgb.mov -vsync 0 ./images/%05d.png