?4 被添加到这个 Bash 脚本的文件名中,为什么?

?4 被添加到这个 Bash 脚本的文件名中,为什么?

我有这个脚本,可以从大量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.mp4video2.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,然后从名称的开头去除所有非数字。

相关内容