我使用 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有两个空格之间^=^,^⟵⟶%%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 /f
1st/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
带有StdOut
ffprobe
findstr
/End of a line
regex
数字([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.的结果if
是true
或者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^,位于行尾,就在换行符处,在执行时,命令解释器会将其视为一行,转义应用的换行符。
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
)
观察:3这
if
也可以在不同的布局中使用,例如: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"
),并在执行时间中我们逐一增加创建/定义从转换中排除的文件的信息:
)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
循环,它将用于回显每个用名称定义的变量_skip_*
,这是通过保存执行过程中忽略的文件来定义的,并且此循环将获取=
(第二/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^=2 ⇄ Skipped 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 文件中最后一个部分的第二部分和最后一部分将让循环发生(隔离),并且仅在列出所有跳过的文件后,它才会暂停/无限超时,等待按下某个键,从而关闭/完成 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
),如果没有文件被忽略if
,只需添加2^>nul
在'set _skip_*2^>nul'
,在最后一个循环内for
:
(for /f tokens^=2^delims^=^= %%i in ('set _skp_ 2^>nul')do echo\%%~i)do...
- 观察4在这里,
echo;
只是为了在列出跳过的文件之前创建一个分隔线。
- 传统/教学布局中的相同代码:
@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