如何使用 FFmpeg 從影片中刪除多個片段?

如何使用 FFmpeg 從影片中刪除多個片段?

我正在嘗試使用 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 秒以結束,然後將它們合併到流out1concat過濾,靜置 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,複製它們的音訊和視訊編解碼器,並產生一個ace.tvshow應該只有 a、c 和 e 部分的檔案。然後記住刪除ace-files.txt, a.tvshow,c.tvshowe.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

相關內容