下面的代码有什么问题?

name='$filename | cut -f1 -d'.''

就像这样,我得到的字面值字符串$filename | cut -f1 -d'。',但如果我删除引号,我什么也得不到。与此同时,打字

"test.exe" | cut -f1 -d'.'

在shell中给出我想要的输出,test。我已经知道$filename已经被分配了正确的值。我要做的是给一个变量分配没有扩展名的文件名。


当前回答

#!/bin/bash
file=/tmp/foo.bar.gz
echo $file ${file%.*}

输出:

/tmp/foo.bar.gz /tmp/foo.bar

注意,只有最后一个扩展被删除。

其他回答

当您希望在script/command中执行命令时,应该使用命令替换语法$(command)。

所以直线是

name=$(echo "$filename" | cut -f 1 -d '.')

代码的解释:

Echo获取变量$filename的值并将其发送到标准输出 然后获取输出并将其输送到cut命令 切割将使用。作为分隔符(也称为分隔符),用于将字符串切割成段,并通过-f选择希望在输出中出现的段 然后$()命令替换将获得输出并返回其值 返回值将被赋给名为name的变量

注意,这给出了到第一个周期的变量部分:

$ filename=hello.world
$ echo "$filename" | cut -f 1 -d '.'
hello
$ filename=hello.hello.hello
$ echo "$filename" | cut -f 1 -d '.'
hello
$ filename=hello
$ echo "$filename" | cut -f 1 -d '.'
hello

如果知道扩展名,可以使用basename

$ basename /home/jsmith/base.wiki .wiki
base

这个包含了所有的可能性!(点在路径上与否;延长或不延长):

Tmp1 =${filename##*/};tmp2=${tmp1:1};filename_noextension=$(echo -n ${Tmp1:0:1};echo ${tmp2%.*})

注:

它提供了没有任何扩展名的文件名。因此$filename_noextension变量中没有路径。 最后得到两个不需要的变量$tmp1和$tmp2。确保没有在脚本中使用它们。

测试示例:

文件名= . bashrc;Echo“filename: $filename”;文件名tmp1 = $ {# # * /}; tmp2 = $ {tmp1:1}; filename_noextension = $ (echo - n $ {tmp1:0:1}; echo $ {tmp2%。*});Echo“没有扩展名的文件名:$filename_noextension”

文件名= .bashrc.txt;Echo“filename: $filename”;文件名tmp1 = $ {# # * /}; tmp2 = $ {tmp1:1}; filename_noextension = $ (echo - n $ {tmp1:0:1}; echo $ {tmp2%。*});Echo“没有扩展名的文件名:$filename_noextension”

文件名= .bashrc.txt.tar;Echo“filename: $filename”;文件名tmp1 = $ {# # * /}; tmp2 = $ {tmp1:1}; filename_noextension = $ (echo - n $ {tmp1:0:1}; echo $ {tmp2%。*});Echo“没有扩展名的文件名:$filename_noextension”

文件名= ~ / . bashrc;Echo“filename: $filename”;文件名tmp1 = $ {# # * /}; tmp2 = $ {tmp1:1}; filename_noextension = $ (echo - n $ {tmp1:0:1}; echo $ {tmp2%。*});Echo“没有扩展名的文件名:$filename_noextension”

文件名= ~ / .bashrc.txt.tar;Echo“filename: $filename”;文件名tmp1 = $ {# # * /}; tmp2 = $ {tmp1:1}; filename_noextension = $ (echo - n $ {tmp1:0:1}; echo $ {tmp2%。*});Echo“没有扩展名的文件名:$filename_noextension”

文件名= bashrc; (Echo“filename: $filename”;文件名tmp1 = $ {# # * /}; tmp2 = $ {tmp1:1}; filename_noextension = $ (echo - n $ {tmp1:0:1}; echo $ {tmp2%。*});Echo“没有扩展名的文件名:$filename_noextension”

文件名= bashrc.txt;Echo“filename: $filename”;文件名tmp1 = $ {# # * /}; tmp2 = $ {tmp1:1}; filename_noextension = $ (echo - n $ {tmp1:0:1}; echo $ {tmp2%。*});Echo“没有扩展名的文件名:$filename_noextension”

文件名= bashrc.txt.tar;Echo“filename: $filename”;文件名tmp1 = $ {# # * /}; tmp2 = $ {tmp1:1}; filename_noextension = $ (echo - n $ {tmp1:0:1}; echo $ {tmp2%。*});Echo“没有扩展名的文件名:$filename_noextension”

文件名= ~ / bashrc; (Echo“filename: $filename”;文件名tmp1 = $ {# # * /}; tmp2 = $ {tmp1:1}; filename_noextension = $ (echo - n $ {tmp1:0:1}; echo $ {tmp2%。*});Echo“没有扩展名的文件名:$filename_noextension”

文件名= ~ / bashrc.txt.tar;Echo“filename: $filename”;文件名tmp1 = $ {# # * /}; tmp2 = $ {tmp1:1}; filename_noextension = $ (echo - n $ {tmp1:0:1}; echo $ {tmp2%。*});Echo“没有扩展名的文件名:$filename_noextension”

正如Hawker65在chepner answer的评论中指出的那样,投票最多的解决方案既不关心多个扩展名(如filename.tar.gz),也不关心路径其余部分的点(如this.path/with.dots/in.path.name)。 一个可能的解决方案是:

a=this.path/with.dots/in.path.name/filename.tar.gz
echo $(dirname $a)/$(basename $a | cut -d. -f1)

我的建议是使用basename。 它在Ubuntu中是默认的,直观的代码简单,可以处理大多数情况。

下面是一些处理空格和多点/子扩展的子案例:

pathfile="../space fld/space -file.tar.gz"
echo ${pathfile//+(*\/|.*)}

它通常从一开始就去掉扩展,但在我们的…路径

echo **"$(basename "${pathfile%.*}")"**  
space -file.tar     # I believe we needed exatly that

这里有一个重要的提示:

我在双引号中使用双引号来处理空格。 单报价将不通过由于短信$。 Bash是不寻常的,由于扩展,读取“第二个”和“第一个”引号。

但是,您仍然需要考虑.hidden_files

hidden="~/.bashrc"
echo "$(basename "${hidden%.*}")"  # will produce "~" !!!  

不是预期的“”结果。使用$HOME或/home/user_path/ 因为bash是“不寻常的”,不要展开“~”(搜索bash bash)

hidden2="$HOME/.bashrc" ;  echo '$(basename "${pathfile%.*}")'