如果我的变量包含带引号的 glob,并且我在不使用双引号的情况下扩展它,为什么 glob 会消失?

如果我的变量包含带引号的 glob,并且我在不使用双引号的情况下扩展它,为什么 glob 会消失?

我试图使用不带引号的字符串扩展将两个参数传递给 tar;第一个是命令行标志--exclude,第二个包含一个*字符。为了避免过早的通配符,我尝试引用*

shopt -s nullglob
x="--exclude '*'"
echo tar $x

让我惊讶的是,'*'完全消失了!这是输出:

tar --exclude

我知道我可以从 POSIX shell 切换到 Bash 并使用数组来避免这个令人头疼的问题。但我仍然想知道:那个 POSIX shell 片段到底是怎么回事?为什么它删除了引用的 glob?我本以为它会保留引用的部分,或者一路扩展并扩展全局。但它两者都没有做到。

答案1

首先,您已经在使用bash- 或者至少是一些非 POSIX 选项。 (POSIX 也不shoptnullglob。)

现在让我们解释一下发生了什么。

x="--exclude '*'"
echo tar $x

该变量$x在空格上进行分词,并将生成的部分用于通配。--exclude没有通配符并且逐字保留。'*'没有匹配项(除非您的文件名字面上以单引号开头和结尾),因此因为它包含通配符并且您已将nullglob其设置为删除。结果是

tar --exclude

既然你已经在用了bash,不妨好好做一下,

x=('--exclude' '*')
echo tar "${x[@]}"

如果您想要 POSIX 解决方案,我能提供的最好的方法是重用主要参数列表:

set -- '--exclude' '*'
echo tar "$@"

答案2

对于像这样的极端情况,shell 扩展的顺序bash 文档通常是意外行为的原因。 Posix 具有或多或少相同的行为,但表述不太清楚。

这是发生的事情:

您已将 nullglob 转为 nullglob,因此失败的 glob 为空。

接下来,您设置一个包含文字星号和单引号的变量。

然后,您回显变量的内容,并且会发生以下情况:

  1. 该变量被扩展为文本。该文本包含一个'*'
  2. '*'尽管您认为单引号只会使其成为字符串文字,但仍会替换任何存在的文件名。Nullglob 将其转换为空字符串
  3. 命令被执行

相关内容