我尝试设置从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++}')"
答案 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
)的输出。