给定一个字符串文件路径,例如/foo/fizzbuzz。酒吧,我如何使用bash提取只是fizzbuzz部分说的字符串?


当前回答

除了这个答案中使用的POSIX符合语法外,

basename string [suffix]

就像在

basename /foo/fizzbuzz.bar .bar

GNU basename支持另一种语法:

basename -s .bar /foo/fizzbuzz.bar

同样的结果。区别和优势在于-s意味着-a,它支持多个参数:

$ basename -s .bar /foo/fizzbuzz.bar /baz/foobar.bar
fizzbuzz
foobar

这甚至可以通过使用-z选项将输出与NUL字节分开来实现文件名安全,例如对于这些包含空格,换行符和glob字符(由ls引用)的文件:

$ ls has*
'has'$'\n''newline.bar'  'has space.bar'  'has*.bar'

读入数组:

$ readarray -d $'\0' arr < <(basename -zs .bar has*)
$ declare -p arr
declare -a arr=([0]=$'has\nnewline' [1]="has space" [2]="has*")

readarray -d要求Bash 4.4或更新版本。对于旧版本,我们必须循环:

while IFS= read -r -d '' fname; do arr+=("$fname"); done < <(basename -zs .bar has*)

其他回答

看看basename命令:

NAME="$(basename /foo/fizzbuzz.bar .bar)"

指示它删除后缀.bar,结果是NAME=fizzbuzz

你可以使用

mv *<PATTERN>.jar "$(basename *<PATTERN>.jar <PATTERN>.jar).jar"

例如:-我想从我的文件名中删除- snapshot。用于下面的命令

 mv *-SNAPSHOT.jar "$(basename *-SNAPSHOT.jar -SNAPSHOT.jar).jar"

使用basename假设您知道文件扩展名是什么,不是吗?

而且我相信各种正则表达式建议不能处理包含多个“”的文件名。

下面的内容似乎可以解决双点问题。哦,还有本身包含“/”的文件名(只是为了好玩)

用帕斯卡尔的话来说,“对不起,这个脚本太长了。我没有时间把它缩短。”


  #!/usr/bin/perl
  $fullname = $ARGV[0];
  ($path,$name) = $fullname =~ /^(.*[^\\]\/)*(.*)$/;
  ($basename,$extension) = $name =~ /^(.*)(\.[^.]*)$/;
  print $basename . "\n";
 
perl -pe 's/\..*$//;s{^.*/}{}'

除了这个答案中使用的POSIX符合语法外,

basename string [suffix]

就像在

basename /foo/fizzbuzz.bar .bar

GNU basename支持另一种语法:

basename -s .bar /foo/fizzbuzz.bar

同样的结果。区别和优势在于-s意味着-a,它支持多个参数:

$ basename -s .bar /foo/fizzbuzz.bar /baz/foobar.bar
fizzbuzz
foobar

这甚至可以通过使用-z选项将输出与NUL字节分开来实现文件名安全,例如对于这些包含空格,换行符和glob字符(由ls引用)的文件:

$ ls has*
'has'$'\n''newline.bar'  'has space.bar'  'has*.bar'

读入数组:

$ readarray -d $'\0' arr < <(basename -zs .bar has*)
$ declare -p arr
declare -a arr=([0]=$'has\nnewline' [1]="has space" [2]="has*")

readarray -d要求Bash 4.4或更新版本。对于旧版本,我们必须循环:

while IFS= read -r -d '' fname; do arr+=("$fname"); done < <(basename -zs .bar has*)