linux – Bash脚本 – 不检测变量中的文件名教程
在BASH脚本中,我正在尝试检测文件是否存在.文件名在变量中,但-e命令似乎无法检测到该文件.以下代码始终输出“〜/ misc / tasks / drupal\_backup.sh不存在”
filename="~/misc/tasks/drupal_backup.sh"
if [ -e "$filename" ]; then
echo "$filename exists"
else
echo "$filename does not exist"
fi
另一方面,以下代码正确检测文件:
if [ -e ~/misc/tasks/drupal_backup.sh ]; then
echo "$filename exists"
else
echo "$filename does not exist"
fi
为什么会这样?当文件名在变量中时,如何让它检测文件?
最佳答案:
这是一个有趣的.将$HOME替换为〜就像删除赋值中的引号一样.
如果你把一个set -x放在那个脚本的顶部,你会看到带引号的版本将filename设置为〜/ …这是给-e的.如果删除引号,则将filename设置为expanded / home / somebody / ….所以在第一种情况下,您会看到:
+ [ -e ~/... ]
并且它不喜欢它.在第二种情况下,您会看到:
+ [ -e /home/somebody/... ]
它确实有效.
如果你在没有变量的情况下这样做,你会看到:
+ [ -e /home/somebody/... ]
而且,是的,它有效.
经过一番调查后,我发现它实际上就是bash执行扩展的顺序.从bash手册页:
The order of expansions is: brace expansion, tilde expansion, parameter, variable and arithmetic expansion and command substitution (done in a left-to-right fashion), word splitting, and pathname expansion.
这就是为什么它不起作用,变量在波浪扩展后被替换.换句话说,在bash想要扩展〜的时候,没有一个.只有在变量扩展之后,这个词才会变成〜/ …并且之后就没有波浪扩展了.
你可以做的一件事是将你的if语句改为:
if [[ -e $(eval echo $filename) ]]; then
这将两次评估$filename参数.第一次(使用eval),在波浪扩展阶段期间将没有〜但是在变量扩展阶段期间$filename将被更改为〜/ ….
然后,在第二次评估(作为if本身的一部分完成的评估)中,〜将在波浪扩展阶段期间存在.
我已经在我的.profile文件上测试了它,它似乎工作,我建议你在你的特定情况下确认.