我有一个这样的脚本:
for chain in http https ssh
do
iptables -nvxL $chain | tail -1 | awk '{print $2}'
done
但我真正想做的是将iptables
每次迭代的命令输出捕获到一个不同的变量中,该变量的名称应该等于 的当前值chain
。
因此,对于第一个循环迭代(哪里$chain
是http
),我希望发生这种情况:
http=$(iptables -nvxL http | tail -1 | awk '{print $2}')
然后接下来我想要这个:
https=$(iptables -nvxL https | tail -1 | awk '{print $2}')
希望你明白了,但不确定如何去做。
在你的情况下,我会为此使用一个关联数组:
然后,您可以通过取消引用来访问输出,如
或者
关联数组的答案几乎可以肯定是您想如何做到这一点。
但是,如果您确实想要命名为
http
、https
和$ 的变量ssh
(您将使用 、 和 $ 引用$http
)$https
,ssh
那么您可以使用printf -v
将命令输出分配给名称包含在 中的变量$chain
:另请注意,根据此答案,您可以使用
typeset
ordeclare
来实现相同的目的:没有
eval
人在做出这个答案时受到伤害。此外,如果您想像 @penguin359 的回答那样从循环中打印变量值,您可以使用bash 变量间接引用来取消引用变量名 - 同样,不需要eval
s :更新:我忘了引用
$2
示例中的嵌入。它现在应该按照 OP 的要求工作。此外,这个答案和使用背后的要点eval
是提供一个跨多个不同实现使用标准 Bourne shell 语法的可移植示例。我已经在 bash、dash 和 zsh 上测试了以下所有代码。Dash 是一个更好的可移植性测试,因为它使用更有限且更符合 Bourne shell 语法。它也是 Debian/Ubuntu 上/bin/sh
. Dash中不存在 和typeset
。printf -v
我确实认为eval
它很丑陋,但它是 Bourne shell 中的第一个机制,是实现真正可移植性的最佳方法。虽然我认为关联数组是可行的方法,但我想为他最初的问题提供一个准确的答案,并且可以移植到除 Bash 之外的其他 Bourne shell。这就是传统上动态变量名称的处理方式:
Bourne shell 中的
eval
运算符用于将变量代入命令,然后在完成后将其作为命令重新运行(重新评估)。在第一次替换该命令后,$chain 变量和第一组引号就完成了,看起来像这样:然后评估为对命令扩展的变量赋值
iptables
。由于内部命令使用单引号,我无法使用它们来引用eval
语句,因此使用"\"
and\""
将其引用为 eval 并根据需要在结果命令中传递双引号和单引号。命令扩展周围的双引号可能不需要,但如果命令扩展为多个单词,我会遇到一些 shell 问题。要转储这些内容,请执行类似的操作:第一个美元符号被引用并通过。这导致在展开循环后执行以下三个命令: