我有这个脚本,可以从大量mp4
视频中提取一帧并将它们存储为png
for i in *.mp4; do
[ -f "$i" ] || break
number=$(echo "$i" | grep -o -E '[0-9]+')
name=frame"$number".png
ffmpeg -i "$i" -ss 00:01:00.000 -vframes 1 "$name"
done
视频的名称类似于video1.mp4
、video2.mp4
等。由于循环将视频打乱顺序,因此我从视频中提取数字以生成输出文件名。
这很好用,但是文件的文件名png
如下:
frame1?4.png
, frame3?4.png
, frame3?4.png
...
为什么?4
?
啊,我现在明白了,4 来自 mp4 上的 4。如何只获取第一个数字?
答案1
这?
只是一个占位符输出,用于ls
表示您嵌入在文件名中的换行符。
echo foo3bar.mp4 | grep -Eo '[0-9]+'
会输出:
3
4
因为该文件名中有两个由 1 个或多个十进制数字组成的序列。
相反,你可以这样做:
for file in *[0-9]*.mp4; do
n=${file%.*} # remove the extension
n=${n%"${n##*[0-9]}"} # remove anything after the last digit
n=${n##*[!0-9]} # remove everything up to the last non-digit.
ffmpeg...
done
这将提取文件名中最右边的十进制数字序列,并去除其扩展名,并避免为每个文件运行多个命令。
请注意,该代码仅使用标准 POSIX sh 语法,因此您不需要bash
运行它。使用bash
或其他ksh93
类似的 shell,如果您知道文件的根名称仅包含一个数字序列,您还可以使用:
n=${file%.*} # remove the extension
n=${n//[^0-9]} # remove all non-digit characters
具有bash
特定功能:
[[ $file =~ ([0-9]+).*\. ]] && n=${BASH_REMATCH[1]}
会提取最左边根名称中的数字序列。
和zsh
:
n=${(SM)${file:r}##<->}
答案2
您已正确识别4
出来自.mp4
文件名中的扩展名。问号可能会ls
显示在每个匹配之间插入的换行符(顺便说一句,grep -o
这是不信任显示的文件名的原因之一)。ls
要提取扩展名之前的第一个数字,或者更确切地说,最后一个数字.mp4
,您可以这样做
n=${i%.mp4}
number=${n##*[!0-9]}
这首先从名称中去除扩展名i
,然后从名称的开头去除所有非数字。