FFmpeg 使用 -ss 参数时损坏视频

FFmpeg 使用 -ss 参数时损坏视频

我正在尝试使用 ffmpeg 提取视频的中间部分。我使用的命令是:

ffmpeg -i input.mp4 -c:a copy -c:v copy -ss 00:04:20 -t 50 output.mp4

我得到的输出存在不同的问题。有时我根本无法播放整个视频。现在我得到的视频断断续续几秒钟(仍然可以听到声音),然后就可以正常播放了。

这是我在 cmd 中得到的全部内容:

E:\Videos>ffmpeg -i input.mp4 -c:v copy -c:a copy -ss 00:04:20 -t 50 output.mp4
ffmpeg version 2022-06-06-git-73302aa193-full_build-www.gyan.dev Copyright (c) 2000-2022 the FFmpeg developers
  built with gcc 11.3.0 (Rev1, Built by MSYS2 project)
  configuration: --enable-gpl --enable-version3 --enable-static --disable-w32threads --disable-autodetect --enable-fontconfig --enable-iconv --enable-gnutls --enable-libxml2 --enable-gmp --enable-bzlib --enable-lzma --enable-libsnappy --enable-zlib --enable-librist --enable-libsrt --enable-libssh --enable-libzmq --enable-avisynth --enable-libbluray --enable-libcaca --enable-sdl2 --enable-libdav1d --enable-libdavs2 --enable-libuavs3d --enable-libzvbi --enable-librav1e --enable-libsvtav1 --enable-libwebp --enable-libx264 --enable-libx265 --enable-libxavs2 --enable-libxvid --enable-libaom --enable-libjxl --enable-libopenjpeg --enable-libvpx --enable-mediafoundation --enable-libass --enable-frei0r --enable-libfreetype --enable-libfribidi --enable-liblensfun --enable-libvidstab --enable-libvmaf --enable-libzimg --enable-amf --enable-cuda-llvm --enable-cuvid --enable-ffnvcodec --enable-nvdec --enable-nvenc --enable-d3d11va --enable-dxva2 --enable-libmfx --enable-libshaderc --enable-vulkan --enable-libplacebo --enable-opencl --enable-libcdio --enable-libgme --enable-libmodplug --enable-libopenmpt --enable-libopencore-amrwb --enable-libmp3lame --enable-libshine --enable-libtheora --enable-libtwolame --enable-libvo-amrwbenc --enable-libilbc --enable-libgsm --enable-libopencore-amrnb --enable-libopus --enable-libspeex --enable-libvorbis --enable-ladspa --enable-libbs2b --enable-libflite --enable-libmysofa --enable-librubberband --enable-libsoxr --enable-chromaprint
  libavutil      57. 26.100 / 57. 26.100
  libavcodec     59. 33.100 / 59. 33.100
  libavformat    59. 24.100 / 59. 24.100
  libavdevice    59.  6.100 / 59.  6.100
  libavfilter     8. 40.100 /  8. 40.100
  libswscale      6.  6.100 /  6.  6.100
  libswresample   4.  6.100 /  4.  6.100
  libpostproc    56.  5.100 / 56.  5.100
Input #0, mov,mp4,m4a,3gp,3g2,mj2, from 'input.mp4':
  Metadata:
    major_brand     : isom
    minor_version   : 512
    compatible_brands: isomiso2avc1mp41
    encoder         : Lavf58.76.100
  Duration: 00:08:26.77, start: 0.000000, bitrate: 4632 kb/s
  Stream #0:0[0x1](und): Video: h264 (High) (avc1 / 0x31637661), yuv420p(tv, bt709, progressive), 1280x720 [SAR 1:1 DAR 16:9], 4432 kb/s, 30 fps, 30 tbr, 15360 tbn (default)
    Metadata:
      handler_name    : VideoHandler
      vendor_id       : [0][0][0][0]
  Stream #0:1[0x2](und): Audio: aac (LC) (mp4a / 0x6134706D), 48000 Hz, stereo, fltp, 191 kb/s (default)
    Metadata:
      handler_name    : SoundHandler
      vendor_id       : [0][0][0][0]
Output #0, mp4, to 'output.mp4':
  Metadata:
    major_brand     : isom
    minor_version   : 512
    compatible_brands: isomiso2avc1mp41
    encoder         : Lavf59.24.100
  Stream #0:0(und): Video: h264 (High) (avc1 / 0x31637661), yuv420p(tv, bt709, progressive), 1280x720 [SAR 1:1 DAR 16:9], q=2-31, 4432 kb/s, 30 fps, 30 tbr, 15360 tbn (default)
    Metadata:
      handler_name    : VideoHandler
      vendor_id       : [0][0][0][0]
  Stream #0:1(und): Audio: aac (LC) (mp4a / 0x6134706D), 48000 Hz, stereo, fltp, 191 kb/s (default)
    Metadata:
      handler_name    : SoundHandler
      vendor_id       : [0][0][0][0]
Stream mapping:
  Stream #0:0 -> #0:0 (copy)
  Stream #0:1 -> #0:1 (copy)
Press [q] to stop, [?] for help
frame= 1414 fps=0.0 q=-1.0 Lsize=   33243kB time=00:00:50.01 bitrate=5444.8kbits/s speed= 545x
video:32014kB audio:1177kB subtitle:0kB other streams:0kB global headers:0kB muxing overhead: 0.158663%

答案1

您应该阅读以下文档:http://trac.ffmpeg.org/wiki/Seeking

基本上,如果您不想重新编码(-c:v copy),ffmpeg 将仅在关键帧上剪切视频流,而这些关键帧通常不在所需的确切时间戳上。这解释了为什么在视频最终显示之前,您只有几秒钟的音频。

至于输出无法播放(损坏?)的情况,我不确定。但选项的行为-ss取决于它是在之前还是之后-i。after-ss表示-i输入流被解码并重新编码,直到最终所需的时间:当它与结合时会发生什么-c:v copy尚不清楚,我不会使用它。

如果您真的不想重新编码,您应该尝试-ss在之前移动-i:开始时您仍然会有一些没有视频的音频,但在我看来它看起来更一致,应该可以修复“损坏”(或任何其他)情况。但如果您真的需要在非常准确的时间提取视频(不限于关键帧),那么您必须重新编码流。

相关内容