使用printf%q传递带空格的参数

时间:2019-03-04 21:46:08

标签: bash printf whitespace

假设我有一个脚本,可以打印出传递给它的参数数量:

# file: num_args

echo "Number of arguments: $#"

现在我的问题与以下调用有关:


> ./num_args a b c
> Number of arguments: 3 # As I would expect.

> ./num_args "a b c"
> Number of arguments: 1 # As I would expect.

> ./num_args a\ b\ c
> Number of arguments: 1 # As I would expect.

> printf "%q\n" "a b c"
> a\ b\ c                # As I would expect.

> ./num_args $(printf "%q" "a b c")
> Number of arguments: 3 # NOT as I would expect.

鉴于printf手册页指出了这一点

 %q     ARGUMENT is printed in a format that can be reused as shell input, escaping non-printable characters with the proposed POSIX $'' syntax.

我不确定在以上最后一种情况下会发生什么。

1 个答案:

答案 0 :(得分:0)

printf %q输出被正确地转义,可以被外壳解释器解析为源代码。

但是,未引用的扩展名不会被解析为源代码。它们仅经历字符串拆分和glob扩展阶段。 (这是一件好事:否则,在shell脚本中处理不受信任的文件名或其他潜在危险的内容几乎是不可能的。)

您可以将此输出替换为shell命令,例如sh -c "...$foo..."eval "...$foo..."ssh somehost "...$foo...";它不能(不应该)不加引号使用。


一些重要的参考文献:

  • BashFAQ #50解释了为什么不能通过将包含引号的字符串展开成无引号来使用它们。
  • BashFAQ #48解释了使用eval(上面讨论的解决方法系列之一)所涉及的风险。