在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文件上测试了它,它似乎工作,我建议你在你的特定情况下确认.

标签: shell, bash, linux, filenames

相关文章推荐

添加新评论,含*的栏目为必填