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

cev*_*ing 8 bash quoting

如何将带空格的字符串传递给一个命令,该命令返回一个带空格的字符串?

我尝试了以下四个版本.

arg='one arg'

arg() { echo "arg: $1"; }

printf '1 |%s|\n' $(arg "$arg")
printf '2 |%s|\n' "$(arg $arg)"
printf '3 |%s|\n' "$(arg \"$arg\")"
printf '4 |%s|\n' "$(arg '$arg')"
Run Code Online (Sandbox Code Playgroud)

他们都失败了:

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

如何得到这个结果?

? |arg: one arg|

Cha*_*ffy 13

语法

使用$()创建新的引用上下文.因此,双引号的命令置换是完全独立的那些以外的它,和闭合的内部外双引号开始一个新的且独立的对.

arg='one arg'

arg() { echo "arg: $1"; }

printf '? |%s|\n' "$(arg "$arg")"
Run Code Online (Sandbox Code Playgroud)

......适当地发出:

? |arg: one arg|
Run Code Online (Sandbox Code Playgroud)

理由(和历史)

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

printf '%s |%s|\n' "$(arg "$(arg "$arg")")"
Run Code Online (Sandbox Code Playgroud)

使用pre-POSIX反引号语法代替$(),您的尝试#3将是正确的:

printf '3 |%s|\n' "`arg \"$arg\"`"
Run Code Online (Sandbox Code Playgroud)

但是,当嵌套深度增加时,需要反斜杠 - 转义引号和嵌套反引号很快就无法工作.只添加一个嵌套arg使它成为:

printf '3a |%s|\n' "`arg \"\`arg \\\"$arg\\\"\`\"`"
Run Code Online (Sandbox Code Playgroud)

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

printf '3b |%s|\n' "`arg \"\`arg \\\"\\\`arg \\\\\\\"$arg\\\\\\\"\\\`\\\"\`\"`"
Run Code Online (Sandbox Code Playgroud)

而现代语法,它只是:

printf '3b |%s|\n' "$(arg "$(arg "$(arg "$arg")")")"
Run Code Online (Sandbox Code Playgroud)

容易,容易.

  • 请注意,语法高亮显示器不知道新的上下文.(或者至少,表现不好.) (3认同)