53

我们需要将大型实时 WMV 视频源拆分为大小相同的小块。我们制作了一个可以正常工作的脚本,除了一件事:视频块不以关键帧开头,因此在播放大多数视频块时,它们不会显示任何图像,直到最终来自原始视频的关键帧到达。

有没有办法告诉 ffmpeg 让输出视频从关键帧开始?

这是我们的命令行现在的样子:

ffmpeg.exe -i "C:\test.wmv" -ss 00:00:00 -t 00:00:05 -acodec copy -vcodec copy -async 1 -y  "0000.wmv"
ffmpeg.exe -i "C:\test.wmv" -ss 00:00:05 -t 00:00:05 -acodec copy -vcodec copy -async 1 -y  "0001.wmv"

等等...

4

6 回答 6

50

FFMPEG 的最新版本包括一个新选项“segment”,这正是我认为您需要的。

ffmpeg -i INPUT.mp4 -acodec copy -f segment -vcodec copy -reset_timestamps 1 -map 0 OUTPUT%d.mp4

这会产生一系列编号的输出文件,这些文件根据关键帧分成几段。在我自己的测试中,它运行良好,尽管我没有在超过几分钟的时间内使用它,而且只使用 MP4 格式。

于 2014-01-30T06:25:07.640 回答
42

正如官方FFMPEG Docs中所述,它对我来说更好地指定-ss timestart before -i input_file.ext,因为它将生成的视频的开头设置(或者我理解)到您指定的时间戳 之前找到的最近的关键帧。

将您的示例更改为:

ffmpeg.exe -ss 00:00:00 -i "C:\test.wmv" -t 00:00:05 -acodec copy -vcodec copy -async 1 -y  "0000.wmv"

我已经测试了这种方法在文件上.flv的工作。.mp4

于 2015-10-17T15:36:44.757 回答
22

这是我可以开始工作的解决方案:

正如 av501 和 d33pika 所建议的,我使用 ffprobe 来查找关键帧的位置。因为 ffprobe 非常冗长,可能需要几秒钟甚至几分钟才能输出所有关键帧,并且无法从冗长的视频中确定我们想要的帧范围,所以我分为 5 个步骤:

  1. 从原始文件中导出一个视频块,大约是所需块大小的两倍。

    ffmpeg -i source.wmv -ss 00:00:00 -t 00:00:06 -acodec copy -vcodec copy -async 1 -y  0001.wmv
    
  2. 使用 ffprobe 查找关键帧的位置。在所需的块大小之后选择最近的关键帧。

    ffprobe -show_frames -select_streams v -print_format json=c=1 0001.wmv
    
  3. 从 ffprobe 的输出中获取该pkt_dts_time关键帧之前的帧。

  4. ffmpeg 在步骤 1 的导出块上,指定相同的输入和输出文件,并指定-ss 00:00:00-t[在步骤 3 中找到的值]。

    ffmpeg -i 0001.wmv -ss 00:00:00 -t 00:00:03.1350000 -acodec copy -vcodec copy -async 1 -y 0001.wmv
    
  5. 在步骤 1 重新开始,使用-ss[在步骤 3 中通过迭代找到的值的累积总和]。

以这种方式进行,我能够有一种有效且强大的方法来在关键帧处分割视频。

于 2012-12-23T18:17:51.200 回答
14

使用较新的构建ffmpeg,可以通过使用ffprobeffmpeg段来实现这一点muxer

  1. 使用ffprobe和 awk 来识别尽可能接近所需块长度的关键帧。
ffprobe -show_frames -select_streams v:0 \
        -print_format csv [SOURCE_VIDEO] 2>&1 |
grep -n frame,video,1 |
awk 'BEGIN { FS="," } { print $1 " " $5 }' |
sed 's/:frame//g' |
awk 'BEGIN { previous=0; frameIdx=0; size=0; } 
{
  split($2,time,".");
  current=time[1];
  if (current-previous >= [DURATION_IN_SECONDS]){
    a[frameIdx]=$1; frameIdx++; size++; previous=current;
  }
}
END {
  str=a[0];
  for(i=1;i<size;i++) { str = str "," a[i]; } print str;
}'

在哪里

  • [SOURCE_VIDEO] = 要分段的视频的路径
  • [DURATION_IN_SECONDS] = 所需的分段长度(以秒为单位)

输出是以逗号分隔的关键帧字符串。

  1. 使用上面的关键帧输出作为ffmpeg.
ffmpeg -i [SOURCE_VIDEO] -codec copy -map 0 -f segment \
       -segment_frames [OUTPUT_OF_STEP_1] [SEGMENT_PREFIX] \
       _%03d.[SOURCE_VIDEO_EXTENSION]

在哪里

  • [SOURCE_VIDEO] = 要分段的视频的路径
  • [OUTPUT_OF_STEP_1] = 逗号分隔的关键帧字符串
  • [SEGMENT_PREFIX] = 段输出的名称
  • [SOURCE_VIDEO_EXTENSION] = 源视频的扩展名(例如,mp4、mkv)
于 2013-07-16T22:58:31.587 回答
6

用于ffprobe -show_frames -pretty <stream>识别关键帧。

于 2012-12-23T14:11:51.423 回答
3

如果您愿意编写一些脚本并希望我以特定的时间间隔进行构图,那么一种方法是

  1. 运行 ffprobe 并从输出中收集 I 帧的位置

    ffprobe -show_streams

  2. 使用相同的脚本运行一系列 -ss -t 命令以获得所需的块。

然后你可以让你的脚本决定最小帧数[假设有两张相距 10 帧的 I 图片,你真的不想在那里分块]。

另一种方法是使用 gstreamer 的 multisfilesink 并将模式设置为关键帧 [但是这将在每个关键帧处分块,因此可能并不理想]

于 2012-12-23T03:36:11.217 回答