如何正确引用嵌套的 sub-shell 参数?

How to quote nested sub-shell arguments correctly?

如何将带空格的字符串传递给 returns 本身就是带空格的字符串的命令?

下面四个版本我都试过了

arg='one arg'

arg() { echo "arg: "; }

printf '1 |%s|\n' $(arg "$arg")
printf '2 |%s|\n' "$(arg $arg)"
printf '3 |%s|\n' "$(arg \"$arg\")"
printf '4 |%s|\n' "$(arg '$arg')"

他们都失败了:

1 |arg:|
1 |one|
1 |arg|
2 |arg: one|
3 |arg: "one|
4 |arg: $arg|

如何得到这个结果?

? |arg: one arg|

语法

使用 $() 创建一个新的引用上下文。因此,双引号 inside 命令替换完全独立于它之外的双引号,并且在关闭外部双引号的内部开始一个新的和独立的对。

arg='one arg'

arg() { echo "arg: "; }

printf '? |%s|\n' "$(arg "$arg")"

...正确发出:

? |arg: one arg|

基本原理(和历史)

使用上述语法,添加额外的嵌套层很容易:

printf '%s |%s|\n' "$(arg "$(arg "$arg")")"

使用 POSIX 之前的反引号语法而不是 $(),您的尝试 #3 应该是正确的:

printf '3 |%s|\n' "`arg \"$arg\"`"

但是,随着嵌套深度的增加,需要对引号和嵌套反引号进行反斜杠转义很快变得不可行。仅添加一个嵌套 arg 即可:

printf '3a |%s|\n' "`arg \"\`arg \\"$arg\\"\`\"`"

添加 两个 层(因此,总共三个 arg 函数调用)更糟糕,让你陷入:

printf '3b |%s|\n' "`arg \"\`arg \\"\\`arg \\\\"$arg\\\\"\\`\\"\`\"`"

而使用现代语法,它只是:

printf '3b |%s|\n' "$(arg "$(arg "$(arg "$arg")")")"

很多,很多更容易。