我试图使用不带引号的字符串扩展将两个参数传递给 tar;第一个是命令行标志--exclude
,第二个包含一个*
字符。为了避免过早的通配符,我尝试引用*
:
shopt -s nullglob
x="--exclude '*'"
echo tar $x
让我惊讶的是,'*'
完全消失了!这是输出:
tar --exclude
我知道我可以从 POSIX shell 切换到 Bash 并使用数组来避免这个令人头疼的问题。但我仍然想知道:那个 POSIX shell 片段到底是怎么回事?为什么它删除了引用的 glob?我本以为它会保留引用的部分,或者一路扩展并扩展全局。但它两者都没有做到。
答案1
首先,您已经在使用bash
- 或者至少是一些非 POSIX 选项。 (POSIX 也不shopt
是nullglob
。)
现在让我们解释一下发生了什么。
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 为空。
接下来,您设置一个包含文字星号和单引号的变量。
然后,您回显变量的内容,并且会发生以下情况:
- 该变量被扩展为文本。该文本包含一个
'*'
'*'
尽管您认为单引号只会使其成为字符串文字,但仍会替换任何存在的文件名。Nullglob 将其转换为空字符串- 命令被执行