我正在尝试使用 FFmpeg 删除视频的几个部分。
例如,想象一下,如果你录制了一个电视节目,然后想剪掉广告。使用 GUI 视频编辑器,这很简单;你只需标记要删除的每个剪辑的开头和结尾,然后选择删除。我正在尝试使用 FFmpeg 从命令行执行同样的事情。
我知道如何将一个片段剪切成新的视频如下:
ffmpeg -i input.avi -ss 00:00:20 -t 00:00:05 -map 0 -codec copy output.avi
这将剪切一段五秒钟的视频并将其保存为新的视频文件,但我怎样才能执行相反的操作并保存整个视频没有指定的剪辑,以及如何指定要删除的多个剪辑?
例如,如果我的视频可以用 ABCDEFG 表示,那么我想创建一个由 ACDFG 组成的新视频。
答案1
嗯,你仍然可以使用trim
过滤。下面是一个例子,假设您想剪掉视频开头和结尾以及中间的三个片段:
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 秒,然后将它们合并到流out1
中concat
过滤,留出 30-40 秒(10 秒)和 50-80 秒(30 秒)。
关于 setpts:我们需要这个,因为 trim 不会修改图片显示时间,并且当我们剪掉 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
答案2
我从来没能使 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 或 0: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
,复制其音频和视频编解码器,并创建一个仅包含 a、c 和 e 部分的文件。ace.tvshow
然后记得删除ace-files.txt
、a.tvshow
和。c.tvshow
e.tvshow
免责声明:我不知道与其他方法相比,这种方法的效率有多低,ffmpeg
但就我的目的而言,这种方法效果更好。希望它能对某些人有所帮助。
答案3
对于那些难以理解 ptQa 方法的人,有一种更简化的方法。无需连接每个步骤,只需在最后完成所有步骤即可。
对于每个输入,定义一个 A/V 对:
//Input1:
[0:v]trim=start=10:end=20,setpts=PTS-STARTPTS,format=yuv420p[0v];
[0:a]atrim=start=10:end=20,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 个输入的完整命令可能如下所示:
ffmpeg -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
答案4
我编写了一个脚本来加快编辑录制的电视节目。该脚本会询问您要保留的片段的开始和结束时间,并将它们拆分成文件。它为您提供了选项,您可以:
- 取一个或多个片段。
- 您可以将各个片段合并为一个结果文件。
- 加入后您可以保留或删除部分文件。
- 您可以保留原始文件或将其替换为新文件。
让我知道你的想法。
#!/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