如何使用FFmpeg从视频中删除多个片段?


51

我正在尝试使用FFmpeg删除视频的一些部分。

例如,想象一下,如果您在电视上录制了一个节目并想剪掉广告。GUI视频编辑器很简单;您只需标记要删除的每个剪辑的开头和结尾,然后选择删除即可。我正在尝试使用FFmpeg从命令行执行相同的操作。

我知道如何将单个片段剪切为视频,如下所示:

ffmpeg -i input.avi -ss 00:00:20 -t 00:00:05 -map 0 -codec copy output.avi

这样会剪切一个五秒钟的剪辑并将其保存为新的视频文件,但是我该如何做相反的操作并在没有指定剪辑的情况下保存整个视频,以及如何指定多个要删除的剪辑呢?

例如,如果我的视频可以由ABCDEFG代表,我想创建一个由ACDFG组成的新视频。



1
这不是我要的,我想将其全部收录在一个“片段”中,但这与原始视频有不同的部分。
Matias 2013年

@Matias,如果您要问如何从视频中剪切出几个片段,然后将其余部分保留原样,那将是一回事,但是您想从中获取一些片段,并将其与其他视频中的片段组合在一起,这不是一个单独的,独特的问题。您必须执行其他问题要求的操作才能获得单独的细分,然后将它们组合在一起。
Synetech

1
@Synetech感谢您的回答。我不想将它们与其他视频的片段结合在一起。我只想从视频中删除一些部分。例如,如果我的视频可以由ABCDEFG代表,我想创建一个由ACDFG组成的新视频。
Matias

@Synetech那不是我,是Tog一定会被误解。
Matias

Answers:


47

好了,您仍然可以使用trim过滤器。这是一个示例,假设您要剪切30-40秒(10秒)和50-80秒(30秒)的片段:

ffmpeg -i in.ts -filter_complex \
"[0:v]trim=duration=30[a]; \
 [0:v]trim=start=40:end=50,setpts=PTS-STARTPTS[b]; \
 [a][b]concat[c]; \
 [0:v]trim=start=80,setpts=PTS-STARTPTS[d]; \
 [c][d]concat[out1]" -map [out1] out.ts

我在这里做什么?我先修剪30秒,40-50秒和80秒以结束,然后将它们out1concat过滤器合并为流。

关于setpts:之所以需要它,是因为修整不会改变图片的显示时间,并且当我们切出10秒时,解码器计数器在这10秒内看不到任何帧。

如果您也想获得音频,则必须对音频流执行相同的操作。因此,命令应为:

ffmpeg -i utv.ts -filter_complex \
"[0:v]trim=duration=30[av];[0:a]atrim=duration=30[aa];\
 [0:v]trim=start=40:end=50,setpts=PTS-STARTPTS[bv];\
 [0:a]atrim=start=40:end=50,asetpts=PTS-STARTPTS[ba];\
 [av][bv]concat[cv];[aa][ba]concat=v=0:a=1[ca];\
 [0:v]trim=start=80,setpts=PTS-STARTPTS[dv];\
 [0:a]atrim=start=80,asetpts=PTS-STARTPTS[da];\
 [cv][dv]concat[outv];[ca][da]concat=v=0:a=1[outa]" -map [outv] -map [outa] out.ts

太好了!为什么需要setpts?您能补充一下说明吗?
拉吉卜

确定,请参阅更新的答案。
ptQa

4
+1确实应该选择答案。它更适用于“多个段”,并且消除了一个接一个地执行多个命令的需要。(除非另一种方法更有效地提高速度)
Knossos 2013年

1
跳过帧时如何维护字幕数据?
Andrew Scagnelli

1
这是非常不人道的。
golopot

15

我永远无法获得ptQa的解决方案,主要是因为我永远无法弄清楚过滤器中的错误意味着什么或如何解决。我的解决方案似乎有点笨拙,因为它可能会留下一团糟,但是如果将其放入脚本中,则可以自动进行清理。我也喜欢这种方法,因为如果第4步出了问题,您将完成第1-3步,因此从错误中恢复效率更高。

基本策略是使用-t-ss获取您想要的每个片段的视频,然后将所有部分结合在一起以形成最终版本。

假设您有6个分段ABCDEF,每个分段的长度为5秒,并且您希望A(0-5秒),C(10-15秒)和E(20-25秒)可以执行以下操作:

ffmpeg -i abcdef.tvshow -t 5 a.tvshow -ss 10 -t 5 c.tvshow -ss 20 -t 5 e.tvshow

要么

ffmpeg -i abcdef.tvshow -t 0:00:05 a.tvshow -ss 0:00:10 -t 0:00:05 c.tvshow -ss 0:00:20 -t 0:00:05 e.tvshow

这将使文件a.tvshow,c.tvshow和e.tvshow。该-t说每个剪辑有多长,所以如果c是30秒长,你可以通过在30或零点00分30秒。该-ss选项说明了跳入源视频的距离,因此它始终与文件的开头有关。

然后,一旦您拥有一堆视频文件,我就制作一个ace-files.txt像这样的文件:

file 'a.tvshow'
file 'c.tvshow'
file 'e.tvshow'

注意开头的“文件”,其后注意转义的文件名。

然后命令:

ffmpeg -f concat -i ace-files.txt -c copy ace.tvshow

将所有文件abe-files.txt合并在一起,复制它们的音频和视频编解码器,并制作一个文件ace.tvshow,该文件应仅位于a,c和e部分。然后,只记得删除ace-files.txta.tvshowc.tvshowe.tvshow

免责声明:我不知道这与其他方法相比在效率上(效率)如何,ffmpeg但出于我的目的,效果更好。希望它可以帮助某人。


4
这个新手对我来说非常容易理解,谢谢
juanpastas 2015年

4
请注意,如果您使用的是bash,则可以避免以下情况创建文件(ace-files.txt在您的示例中):ffmpeg -f concat -i <(printf "file '%s'\n" $(pwd)/prefix_*.tvshow) -c copy output.tvshow
bufh

这确实很有帮助,但是我不得不从文件名中删除单引号,否则它不起作用。
tchaymore

连接视频时,它们之间有一段简短的黑色视频和静音音频,大约25毫秒。有什么想法可以摆脱吗?
34,安东尼奥·博尼法蒂

嗯...我只是检查一下时间偏移和跳跃的数学运算,并确保您没有留下25ms的间隔...我还没有经历过。
xbakesx

5

我编写了一个脚本来加快编辑录制的电视的速度。该脚本会询问您要保留的段的开始和结束时间,并将其拆分为文件。它为您提供了选择,您可以:

  • 采取一个或多个细分。
  • 您可以将这些段合并为一个结果文件。
  • 加入后,您可以保留或删除零件文件。
  • 您可以保留原始文件或将其替换为新文件。

视频实战:这里

让我知道你的想法。

 #!/bin/bash
/bin/date >>segmenter.log

function segment (){
while true; do
    echo "Would you like to cut out a segment ?"
    echo -e "1) Yes\n2) No\n3) Quit"
    read CHOICE
    if [ "$CHOICE" == "3" ]; then
        exit
    elif [ "$CHOICE" == "2" ]; then
        clear
        break
    elif [ "$CHOICE" == "1" ]; then
        clear
        ((segments++))
        echo "What time does segment $segments start ?"
        read SEGSTART
        clear
        echo -e "Segment $segments start set to $SEGSTART\n"  
        echo "What time does segment $segments end ?"
        read SEGEND
        clear
        echo -e "Segment $segments end set to $SEGEND\n"
        break
    else
        clear
        echo -e "Bad option"
        segment "$segments"
    fi
done
if [ "$CHOICE" == "1" ]; then
    echo "Cutting file $file video segment $segments starting at $SEGSTART and ending at $SEGEND"
    ffmpeg -i "$file" -ss $SEGSTART -to  $SEGEND -map 0:0 -map 0:1 -c:a copy -c:v copy  "$filename-part$segments.$extension"  >> segmenter.log 2>&1
    clear
    echo -e "Cut file $filename-part$segments.$extension starting at $SEGSTART and ending at $SEGEND\n"                             
    segment "$segments"
fi
}

file="$1"
filename="${file%.*}"
extension="${file##*.}"
clear
segments=0
segment "$segments"
clear
if (("$segments"==1)); then
mv $filename"-part1."$extension "$filename-segmented.$extension"
elif (("$segments">1)); then
echo "Would you like to join the segments into one file ?"      
       OPTIONS="Yes No Quit"
       select opt in $OPTIONS; do
       clear
        if [ "$opt" == "Quit" ]; then
            exit
        elif [ "$opt" == "Yes" ]; then
            clear
            echo "Joining segments"
            ffmpeg -f concat -i <(for f in $filename"-part"*$extension;         do echo "file '$(pwd)/$f'"; done) -c:a copy -c:v copy "$filename-segmented.$extension" >>         segmenter.log 2>&1
            clear
            echo "Would you like to delete the part files ?"
            select opt in $OPTIONS; do
            clear
            if [ "$opt" == "Quit" ]; then
                exit
            elif [ "$opt" == "Yes" ]; then
                for f in $filename"-part"*$extension; do rm $f; done
                break
            elif [ "$opt" == "No" ]; then
                break
            else
                clear
                echo -e "Bad option\n"
            fi
            done
            break
        clear
        elif [ "$opt" == "No" ]; then
            exit
        else
            clear
            echo -e "Bad option\n"
        fi
    done
fi
echo "Would you like to replace the original file with the result of your changes ?"
OPTIONS="Yes No Quit"
select opt in $OPTIONS; do
    clear
    if [ "$opt" == "Quit" ]; then
        exit
    elif [ "$opt" == "Yes" ]; then
        rm $file
        mv "$filename-segmented.$extension" $file
        break
    elif [ "$opt" == "No" ]; then
        break
    else
        clear
        echo -e "Bad option\n"
    fi
done

2
这个脚本很棒!由于您使用串联的绝对路径,因此仅需进行一次修改:add safe=0,即ffmpeg -f concat -safe 0 -i ...参见ffmpeg.org/ffmpeg-all.html#Options-36
pkSML

2

尽管ptQa提供的答案似乎可行,但我开发了另一种解决方案,事实证明工作正常。

本质上,我要做的是针对要包含在结果中的原始视频的每个部分剪切一个视频。稍后,我将它们与此处解释的Concat Demuxer连接起来。

该解决方案与我首先尝试并提出同步问题的方法相同。我添加的是生成不同视频时的命令-avoid_negative_ts 1。使用此解决方案,同步问题消失了。


1

对于那些在使用ptQa的方法时遇到麻烦的人,可以采用一种更为简化的方法来解决。而不是紧紧抓住每一步,只需要在最后全部完成。

为每个输入定义一个A / V对:

//Input1:
[0:v]trim=start=10:end=20,setpts=PTS-STARTPTS,format=yuv420p[0v];
[0:a]atrim=start=10:end=10,asetpts=PTS-STARTPTS[0a];
//Input2:
[0:v]trim=start=30:end=40,setpts=PTS-STARTPTS,format=yuv420p[1v];
[0:a]atrim=start=30:end=40,asetpts=PTS-STARTPTS[1a];
//Input3:
[0:v]trim=start=30:end=40,setpts=PTS-STARTPTS,format=yuv420p[2v];
[0:a]atrim=start=30:end=40,asetpts=PTS-STARTPTS[2a];

根据需要定义尽可能多的对,然后一次通过将它们全部合并,其中n =总输入计数。

[0v][0a][1v][1a][2v][2a]concat=n=3:v=1:a=1[outv][outa] -map [outv] -map [outa] out.mp4

这可以很容易地循环构造。

使用2个输入的完整命令可能如下所示:

 -y -i in.mp4 -filter_complex 
[0:v]trim=start=10.0:end=15.0,setpts=PTS-STARTPTS,format=yuv420p[0v];
[0:a]atrim=start=10.0:end=15.0,asetpts=PTS-STARTPTS[0a];
[0:v]trim=start=65.0:end=70.0,setpts=PTS-STARTPTS,format=yuv420p[1v];
[0:a]atrim=start=65.0:end=70.0,asetpts=PTS-STARTPTS[1a];[0v][0a][1v]
[1a]concat=n=2:v=1:a=1[outv][outa] -map [outv] -map [outa] out.mp4
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.