5

我想使用avconv在指定时间从视频文件中获取单个图像。

我已经阅读了足够多的关于libav的内容,以为我知道自己在做什么,但还不足以真正知道。

我试过了:

avconv -ss 00:00:01.786 -r 25 -i input_video.h264 -frames 1 output_image.jpg

以及使用 t 来避免 'frames' 参数:

avconv -ss 00:00:01.786 -r 25 -i input_video.h264 -t 0.01 output_image.jpg

并在几秒钟内传递而不是使用 hh:mm:ss.xxx 格式:

avconv -ss 1.786 -r 25 -i input_video.h264 -t 0.01 output_image.jpg

我看到的是当 ss 设置为 0(即“0”、“00:00:00.000”、“0.0”等)时,output_image 只是视频的第一帧。正如预期的那样。

ss 的任何其他值——即使是 0.0001——都会给我视频的最后一帧。

我正在使用来自 Raspbian wheezy repo 的最新 avconv。这种行为对我来说就像一个错误,但我不知道视频流的其他复杂性足以确定。

有谁知道我做错了什么?



额外的问题:我实际上想从同一个视频中获得一大堆这些图像。将命令串在一起似乎以前可以工作,即:

avconv -ss 1.786 -r 25 -i input_video.h264 -t 0.01 output_image1.jpg 
-ss 3.454 -r 25 -i input_video.h264 -t 0.01 output_image2.jpg
-ss 5.823 -r 25 -i input_video.h264 -t 0.01 output_image3.jpg
-ss etc,etc.

但我切换回单个图像来调试这个问题。假设第一个问题得到解决,这是构建此命令的最佳方法还是有更好的方法?

4

2 回答 2

3

编辑:正如 Mulvya 所指出的,ffmpeg 将始终使用 -ss 准确搜索。这个问题是 avconv 独有的。

如果您在-ss选项之前指定选项,-i则 avconv 将不准确地计算帧位置。

您需要首先指定该-i选项,以便 avconv 知道它需要通过流“寻找”,直到它准确地找到正确的时间戳。

此外,您的示例时间与您在示例中指定1.786的每秒帧数不一致。-r 25

因为1/25=0.04任何值-ss都应该可以被除以0.04精确地指定一个单独的框架。

以下内容应获得视频的第 46 帧:

avconv -i input_video.h264 -r 25 -ss 1.8 -frames:v 1 output_image.jpg

如果你想通过它的索引获得一个特定的框架,那么你需要使用bc

avconv -i input_video.h264 -r 25 -ss 0$(echo "scale=2;1000/25" | bc -l) -frames:v 1 output_image.jpg

1000视频的第 1001 帧在哪里(因为0/25是第 1 帧)。

请注意,与@hamboy75 的示例不同,我们将-l(小写 L)传递给,bc以便它执行浮点计算(并且不会四舍五入到最接近的整数)。scale=2用于生成精确到 2dp 的数字。

另请注意,它bc具有输出小于 1 的数字而没有前导零(即.04avconv 不理解)的“特征”。因此,我们还需要在计算前插入一个前导零0$()

使用此命令时,您将获得如下所示的输出

frame=    0 fps=  0 q=0.0 size=       0kB time=10000000000.00 bitrate=   0.0kbit
frame=    0 fps=  0 q=0.0 size=       0kB time=10000000000.00 bitrate=   0.0kbit
frame=    0 fps=  0 q=0.0 size=       0kB time=10000000000.00 bitrate=   0.0kbit
frame=    0 fps=  0 q=0.0 size=       0kB time=10000000000.00 bitrate=   0.0kbit
frame=    0 fps=  0 q=0.0 size=       0kB time=10000000000.00 bitrate=   0.0kbit

这是因为 avconv 正在通过文件查找您准确请求的特定帧。因此,具有较大索引的帧将需要更长的时间来提取,因为 avconv 将始终从流的开头“寻找”流。

因此,可能更希望提取一系列帧:

avconv -i input_video.h264 -r 25 -ss 0$(echo "scale=2;7500/25" | bc -l) -t 0$(echo "scale=2;250/25" | bc -l) output_image_%04d.jpg

此示例从视频中的 5 分钟中提取 10 秒的帧。当然你也可以只使用:

avconv -i input_video.h264 -r 25 -ss 300.0 -t 10.0 | bc -l) output_image_%04d.jpg

但是,请记住,对于任何小于一秒的持续时间,该值应该可以被视频的帧速率整除(即0.0425 fps)。

每个帧的图像将命名为output_image_0001.jpgoutput_image_0002.jpgoutput_image_0003.jpg等。如果您希望对提取的帧执行图像比较,您可能需要考虑使用pngoverjpg以获得更高的保真度。

请注意,如果您指定的帧索引大于视频中存在的帧数,avconv 将简单地提取它找到的最后一帧。您可能希望通过Duration:查看avconv -i input_video.h264.

于 2016-03-04T11:01:53.177 回答
-1

在我的情况下,我想要前 3 分钟每 60 秒的图像...

对于 {0..3} 中的 i ;做 avconv -ss echo $i*60.0 | bc-i input_video.h264 -frames:v 1 -y out$i.jpg ;完毕

如果您在某些时候需要它,您可以稍微更改一下:9。

于 2016-01-07T09:23:32.310 回答