在bash -c中设置变量

时间:2015-03-01 12:52:03

标签: linux bash awk ifconfig

我尝试设置从ifconfig获取接口IP地址的变量,然后再读取它。但是当我执行echo命令时,变量仍然是空的。请看我的代码:

/usr/bin/bash -c "HOST_IPS=$(/usr/bin/ifconfig | /usr/bin/awk 'BEGIN {cnt=0} {if($0 ~ /inet / && cnt==1) {print $2} cnt++}'); echo $HOST_IPS"

但/ bin / echo使用相同的命令工作正常:

/usr/bin/bash -c "echo $(/usr/bin/ifconfig | /usr/bin/awk 'BEGIN {cnt=0} {if($0 ~ /inet / && cnt==1) {print $2} cnt++}')"

3 个答案:

答案 0 :(得分:7)

您必须在最终的echo命令中转义$符号,否则在生成子shell之前,变量$HOST_IPS将被替换为命令字符串:

/usr/bin/bash -c "HOST_IPS=$(/usr/bin/ifconfig | /usr/bin/awk 'BEGIN {cnt=0} {if($0 ~ /inet / && cnt==1) {print $2} cnt++}'); echo \$HOST_IPS"

为了更直接的可见性:

#                                                  v-- insert backslash here
/usr/bin/bash -c "HOST_IPS=$(same as before); echo \$HOST_IPS"

与@ gniourf_gniourf的评论相反,实际上没有必要逃避其他美元符号。但是,如上所述,命令替换不是由子shell(!)执行的;其结果将替换为传递给子shell的命令字符串。电话

mypid() { echo $$; }
bash -c "pid=$(mypid); echo \$pid; mypid"

演示它的工作方式:它将一次打印父shell的PID,并且一旦抱怨mypid不是已知命令,因为子shell不知道该函数。

由于在父shell中运行ifconfig | awk命令不太可能成为问题,因此您可以将命令替换部分保持不变。如果命令由子shell运行很重要,那么你也必须逃避那里的所有事情。

答案 1 :(得分:0)

使用/usr/bin/bash启动子shell。 shell完成后,shell中的所有设置都将丢失 以下子设置在子shell中并在被回显之前丢失:

/usr/bin/bash -c sub=1; echo $sub

是否要在子shell中设置变量,使用stdout传输值:

sub=$(/usr/bin/bash -c "echo 1"); echo $sub

答案 2 :(得分:0)

从您的问题和示例来看,您手头的任务不需要您离开当前环境,因此您无需启动新环境并担心数据丢失。

HOST_IPS=();
while read -r;
  do [[ $REPLY =~ "inet "([^/]*) ]] && HOST_IPS+=(${BASH_REMATCH[1]});
done < <( ip -f inet address show )

如果您希望维护接口IP地址列表,可以在不调用ip的情况下处理当前shell中ifconfig(或awk)的输出。