ffmpeg - ビットレートが一定量より低い場合はファイルをスキップする

ffmpeg - ビットレートが一定量より低い場合はファイルをスキップする

私は、ffmpeg を使用して、さまざまなストリーミング サイトからダウンロードしたビデオを hevc にエンコードします。Windows では、バッチ ファイルを使用してこれらのファイルを変換します。

FOR /F "tokens=*" %%G IN ('dir /b *.mp4') DO ffmpeg -n -i "%%G" -c:v libx265 -crf 22 -c:a libopus -b:a 48k -vbr on -compression_level 10 -frame_duration 60 -application audio "%%~nG.mkv"

これらのファイルの中にはビットレートが非常に低いものがあり、それらには触れたくありません。ffmpeg でこれらのファイルをスキップする方法はありますか? または、ffprobeビットレートを取得してコマンドを使用してスキップするなど、バッチ ファイルに含めることができるコマンドはありますか?

答え1

@echo off 

cd /d "%~dp0" && setlocal enabledelayedexpansion
set "_ffmpeg=F:\2020-SU\Q1569837\ffmpeg\bin\ffmpeg.exe"
set "_ffprobe=F:\2020-SU\Q1569837\ffmpeg\bin\ffprobe.exe"

for %%# in (*.mp4)do for /f tokens^=2^,6^delims^=^,^  %%i in (
'2^>^&1 "!_ffprobe!" -show_entries stream^=bit_rate "%%~f#" ^| "%__APPDIR__%findstr.exe" /e [0-9].kb/s
')do if %%~j gtr 349 2>&1 ("!_ffmpeg!" -y -i "%%~f#" -hide_banner -v error -stats -c:v libx265 -crf 22 ^
     -c:a libopus -b:a 48k -vbr on -compression_level 10 -frame_duration 60 -application audio "%%~n#.mkv" 
     )else set/a "_c+=1+0" && <con: call set "_skp_!_c!=Skipped File: %%~nx# Duration: %%~i Bit Rate: %%~j"
      
echo;& (for /f tokens^=2^delims^=^= %%i in ('set _skp_ 2^>nul')do echo\%%~i) & %__APPDIR__%timeout.exe -1 & endlocal
  • 出力:
x265 [info]: HEVC encoder version 3.4+2-73ca1d7be377
x265 [info]: build info [Windows][GCC 9.3.1][64 bit] 8bit+10bit
x265 [info]: using cpu capabilities: MMX2 SSE2Fast LZCNT SSSE3 SSE4.2 AVX FMA3 BMI2 AVX2
x265 [info]: Main profile, Level-3.1 (Main tier)
x265 [info]: Thread pool created using 4 threads
x265 [info]: Slices                              : 1
x265 [info]: frame threads / pool features       : 2 / wpp(12 rows)
x265 [info]: Coding QT: max CU size, min CU size : 64 / 8
x265 [info]: Residual QT: max TU size, max depth : 32 / 1 inter / 1 intra
x265 [info]: ME / range / subpel / merge         : hex / 57 / 2 / 3
x265 [info]: Keyframe min / max / scenecut / bias  : 23 / 250 / 40 / 5.00
x265 [info]: Lookahead / bframes / badapt        : 20 / 4 / 2
x265 [info]: b-pyramid / weightp / weightb       : 1 / 1 / 0
x265 [info]: References / ref-limit  cu / depth  : 3 / off / on
x265 [info]: AQ: mode / str / qg-size / cu-tree  : 2 / 1.0 / 32 / 1
x265 [info]: Rate Control / qCompress            : CRF-22.0 / 0.60
x265 [info]: tools: rd=3 psy-rd=2.00 early-skip rskip mode=1 signhide tmvp
x265 [info]: tools: b-intra strong-intra-smoothing lslices=4 deblock sao
frame= 1440 fps= 55 q=29.8 Lsize=    2570kB time=00:01:00.11 bitrate= 350.3kbits/s speed=2.32x
x265 [info]: frame I:      6, Avg QP:22.93  kb/s: 1138.86
x265 [info]: frame P:    705, Avg QP:25.55  kb/s: 498.87
x265 [info]: frame B:    729, Avg QP:28.95  kb/s: 98.52
x265 [info]: Weighted P-Frames: Y:0.9% UV:0.6%
x265 [info]: consecutive B-frames: 58.8% 11.0% 11.4% 6.6% 12.2%

encoded 1440 frames in 25.96s (55.47 fps), 298.86 kb/s, Avg QP:27.26

Skipped File: Live_TV_-_Bloomberg.mp4 Duration: 00:00:36.42 Bit Rate: 315
Skipped File: HVDC Light - ABB 3D.mp4 Duration: 00:03:32.16 Bit Rate: 336

観測: 1がある2つのスペース^=^,^⟵⟶%%iで: delims^=^,^spacespace%%i

for %%# in (*.mp4)do for /f tokens^=2^,6^delims^=^,^spacespace%%i in (...


1.- 宿題: 以下の変数をシナリオに合わせて置き換え、bat フォルダーにも移動します。

set "_ffmpeg=F:\2020-SU\Q1569837\ffmpeg\bin\ffmpeg.exe"
set "_ffprobe=F:\2020-SU\Q1569837\ffmpeg\bin\ffprobe.exe"

cd /d "%~dp0"

rem :: if your *.pm4 files are not in the same directory 
rem :: as your bat file, use the full path to drive/folder 
rem :: Example for drive D: folder/subfolder \Media\MP4\Convert 

cd /d "D:\Midia\MP4\Convet"

2.このバッチは複数の を使用していますfor loop。これを機能させるには、Deleyed Expansion実行時に変数が更新/拡張された値を受け取るように を有効にする必要があります。

Setlocal EnableDelayedExpansion

3.残念ながら、あなたの現在のfor /f ... dir .mp4 ...あまり役に立たないので、forすべてを取得する単純なものに置き換えてください.mp4ループ内にリストされます:

for %%# in (*.mp4)do ....

4.mp4ファイルのフルパス/名前( )を取得したループ変数( )を追加で使用し、このループを既に定義している( で説明)への入力として渡します。for /f1st/for/var==%%#%%~f#ffprobe項目5。)、そのコマンドで取得されるトークンと区切り文字。

for /f tokens^=2^,6^delims^=^,^  %%i in (ffmprobe ...  %%~f# ... 

5.ffprobeループ内で使用されるコマンドは次のとおりです。for /f

..\ffprobe.exe -show_entries stream=bit_rate "Google Chrome - Now Everywhere.mp4"

6.スイッチを使用してフィルタリングする出力をStdErrリダイレクトすることから始めますStdOutffprobefindstr/End of a lineregex数字([0-9])を文字列に連結し.kb/s、適切なスケープ処理を使用してforループ:

2^>^&1 "!_ffprobe!" -show_entries stream^=bit_rate "%%~f#" ^| "%__APPDIR__%findstr.exe" /e [0-9].kb/s

7。上記の拡張コマンドをエスケープなしで実行すると、結果は次のようになります。

2>&1 ..\ffprobe.exe -show_entries stream=bit_rate "Google Chrome - Now Everywhere.mkv" | "%__APPDIR__%findstr.exe" /e [0-9].kb/s

8.フィルターによって処理された上記のコマンドの出力はfindstr次のようになります。

再生時間: 00:01:00.08、開始: -0.007000、ビットレート: 350 kb/s

9.フィルターによって処理された上記のコマンドの出力はfindstr次のようになります。

再生時間: 00:01:00.08、開始: -0.007000、ビットレート: 350 kb/s

10.複数の区切り文字を使用すると、%%iそして%%j出力は00:01:00.08そして350: は最後のコマンド出力で、00:01:00.08そして350:

... for /f tokens^=2^,6^delims^=^,^space %%i in (...
Duration: 00:01:00.08, start: -0.007000, bitrate: 350 kb/s

ここに画像の説明を入力してください

11.限界値が350包括的) の場合は、if作業部分でいくつかのオプションを使用する必要があります。

 if %%~j > Bit_Rate      ∕∕ the same:  if %%~j > 349  (349 exclusive) 
 if %%~j  Bit_Rate      ∕∕ the same:  if %%~j  350  (350 inclusive) 

 set "_bit_rate=349"
 if %%~j > %_bit_rate%   ∕∕ the same:  if %%~j > 349  (349 exclusive) 

 set "_bit_rate=350"
 if %%~j  %_bit_rate%   ∕∕ the same:  if %%~j  350  (350 inclusive) 


 if LSS - Less Than                if [integer or int(var)]  < [integer or int(var)]
 if GTR - Greater Than             if [integer or int(var)]  > [integer or int(var)]
 if LEQ - Less Than or Equals      if [integer or int(var)]   [integer or int(var)]
 if GEQ - Greater Than or Equals   if [integer or int(var)]   [integer or int(var)]

12.結果としてiftrueまたはfalse、そしてケースに応じてアクションを実行します。教育目的のため、現在のファイルをtrue場合:

 if %%~j GTR 349 (
      case true
      ffmpeg transcode file mp4
    ) else (
      case false
      skip this file .mp4
      save the full path name 
    )
  •  if %%~j gtr 349 2>&1 ("!_ffmpeg!" -y -i "%%~f#" -hide_banner -v error -stats -c:v libx265 -crf 22 ^
       -c:a libopus -b:a 48k -vbr on -compression_level 10 -frame_duration 60 -application audio "%%~n#.mkv"
  • 観測:2キャラクター:space^は、行末の改行のすぐそばにあり、実行時にコマンド インタープリターはそれを 1 行として扱い、適用された改行をエスケープします。

13.ビットレートが低いファイルの場合、falseの事例ifコマンドでは、除外されたファイルを保存するアクションがあります。ffmpeg変換され、実行の最後にリストされます。

 if %%~j GTR 349 (
      case true
      ffmpeg transcode file mp4
    ) else (
      case false
      skip this file .mp4
      save the full path name
    )
  • 観測:3if次のようなさまざまなレイアウトでも動作します:

    if %%~j GTR 349 (case true
       ffmpeg transcode file mp4 ) else ( 
        case false 
        skip this file .mp4 
        save the full path name 
    ) 
    if %%~j GTR 349 (case true && ffmpeg transcode file mp4 ) else (
        case false && skip this file .mp4 && save the full path name )
    if %%~j GTR 349 (case true && ffmpeg transcode file mp4
        )else case false && skip this file .mp4 && save the full path name 

14.の値を使用して%%~f#%%~iそして%%~j変数は、それぞれ現在のファイルのパスとフルネーム、その継続時間、そのビットレートです。カウンタ ( set/a "_c+=1+0") を簡単に追加でき、実行時に増分して、変換から除外されるファイルの情報を 1 つずつ作成/定義します。

)else set/a "_c+=1+0" && <con: call set "_skp_!_c!=Skipped File: %%~nx# Duration: %%~i Bit Rate: %%~j"

%%~f# == Live_TV_-_Bloomberg.mp4
%%~i  == 00:00:36.42
%%~j  == 315

set "_c+=1+0" && call set "_skp_1=Skipped File: Live_TV_-_Bloomberg.mp4 Duration: 00:00:36.42 Bit Rate: 315"

15.setコマンドは変数や値のリストにも使用でき、set user、定義されているすべての変数user+(strings)次のようにリストされます:

>set USER
USERDOMAIN=LAME_SLUG
USERDOMAIN_ROAMINGPROFILE=LAME_SLUG
USERNAME=ecker
USERPROFILE=C:\Users\ecker

16.最後の行にはfor /fループでは、nameで定義された各変数をエコーするために使用されます_skip_*は、実行中に無視されたファイルを保存することによって定義され、このループは、の記号の後に来るすべてのものを取得します。=(2位/tokens^=2):

for /f tokens^=2^delims^=^= ... set _skp_1 .... echo\%%~i
 _skp_1=Skipped File: Live_TV_-_Bloomberg.mp4 Duration: 00:00:36.42 Bit Rate: 315
       
tokens^=2Skipped File: Live_TV_-_Bloomberg.mp4 Duration: 00:00:36.42 Bit Rate: 315
echo;& (for /f tokens^=2^delims^=^= %%i in ('set _skp_')do echo\%%~i)...

17.bat ファイルの最後の 2 番目と最後の部分では、ループが (分離して) 発生し、スキップされたすべてのファイルをリストした後にのみ、一時停止/無期限のタイムアウトが発生し、何らかのキーが押されるのを待機し、setlocal を閉じて終了し、実行を終了します。

1st part: (for /f .....)
2nd part: %__APPDIR__%timeout.exe -1 & endlocal 

echo; & (for /f tokens^=2^delims^=^= %%i in ('set _skp_')do echo\%%~i) & %__APPDIR__%timeout.exe -1 & endlocal

18.エラーメッセージを回避するには(Environment variable _skip_ not defined)、によってファイルが無視されなかった場合if2^>nul'set _skip_*2^>nul'、最後のループ内for:

(for /f tokens^=2^delims^=^= %%i in ('set _skp_ 2^>nul')do echo\%%~i)do...
  • 観測4ここでは、echo;スキップされたファイルをリストする前に、1 つの分離行を作成するためだけに使用します。


  • 同じコードを従来型/教訓的なレイアウトで表示した場合:
@echo off 

cd /d "%~dp0"
setlocal enabledelayedexpansion

set "_ffmpeg=F:\2020-SU\Q1569837\ffmpeg\bin\ffmpeg.exe"
set "_ffprobe=F:\2020-SU\Q1569837\ffmpeg\bin\ffprobe.exe"

for %%# in (*.mp4) do (

     for /f "tokens=2,6 delims=, " %%i in ('2^>^&1 "!_ffprobe!" -show_entries stream^=bit_rate "%%~f#" ^| "%__APPDIR__%findstr.exe" /e [0-9].kb/s') do (
         
         if %%~j gtr 3200 (
         
             2>&1 "!_ffmpeg!" -y -i "%%~f#" -hide_banner -v error -stats -c:v libx265 -crf 22 -c:a libopus -b:a 48k -vbr on -compression_level 10 -frame_duration 60 -application audio "%%~n#.mkv" 
             
           ) else (
             
             set /a "_c+=1+0"
             set "_skp_!_c!=Skipped File: %%~nx# Duration: %%~i Bit Rate: %%~j"
                 
           )
      )
    
  )

echo.
for /f "tokens=2 delims==" %%i in ('2^>nul set _skp_')do echo\%%~i

%__APPDIR__%timeout.exe -1 
endlocal

関連情報