5

我正在编写一个脚本来收集一些各种网络统计信息。我要做的是从 netstat -i 命令生成一些增量数据。

我正在使用以下 bash 代码收集所需的数据:

declare -a array
n=0
netstat -i | tail -n +3 | while read LINE; do
        echo "Setting array[$n] to $LINE"
        array[$n]=$LINE
        echo "array now have ${#array[@]} entries"
        let n=$n+1
done
echo "array now have ${#array[@]} entries"

该命令的输出是:

Setting array[0] to eth0       1500 0   4946794      0      0 0       2522971      0      0      0 BMRU
array now have 1 entries
Setting array[1] to lo        16436 0     25059      0      0 0         25059      0      0      0 LRU
array now have 2 entries
Setting array[2] to vmnet1     1500 0         6      0      0 0          1126      0      0      0 BMRU
array now have 3 entries
Setting array[3] to vmnet8     1500 0       955      0      0 0          1054      0      0      0 BMRU
array now have 4 entries
Setting array[4] to wlan0      1500 0    613879      0      0 0        351194      0      0      0 BMU
array now have 5 entries
array now have 0 entries

如您所见,数组实际上在 while 循环后消失了,我不明白为什么。

4

3 回答 3

5

每当您使用管道时,您都会创建一个隐式子外壳。当该子shell 终止时,它的变量也会终止。对此的快速解决方法是不要将内容传输到read. 您可以使用进程替换来完成上述操作:

while read LINE; do
        echo "Setting array[$n] to $LINE"
        array[$n]=$LINE
        echo "array now have ${#array[@]} entries"
        let n=$n+1
done < <(netstat -i | tail -n +3)

一种更符合 POSIX 的方法(阅读:更便携,更少 bashist)是让一切都发生在子 shell 中:

netstat -i | tail -n +3 | {
    declare -a array
    n=0
    while read LINE; do
        echo "Setting array[$n] to $LINE"
        array[$n]=$LINE
        echo "array now have ${#array[@]} entries"
        let n=$n+1
    done
    echo "array now have ${#array[@]} entries"
}

您可以在Greg Wooledge 的 wiki阅读此(以及更多)的要点。

于 2012-12-07T12:47:50.343 回答
2

如果您唯一的目标是将命令的输出放入数组(按行),您最好使用(遗憾的是不是很知名)mapfilebash 内置,它是迄今为止最有效的(并且最适合代码高尔夫,计算我与其他可能性相比有多少笔画):

mapfile -t array < <(netstat -i | tail -n +3)

其他答案解释了为什么您的构造不起作用(管道位于子外壳中等等)。

help mapfile有关该命令的所有详细信息和可能性。

于 2012-12-07T14:33:09.023 回答
1

好的,你准备好了吗?

有如何netstat -i | tail -n +3在数组的ba​​sh关联数组中进行转换:

declare -A AANET
while read -a line ;do
    declare -a AI$line
    eval "AI$line=(${line[@]})"
    AANET[$line]=AI$line
  done < <(
    netstat -i |
       tail -n +3)

比现在:

echo ${!AANET[@]}
venet0 eth1 eth0 lo br0

echo ${AANET[eth0]}
AIeth0

对于sub-associative,我们必须使用eval

eval echo \${${AANET[eth0]}[@]}
eth0 1500 0 17647 0 0 0 35426 0 0 0 BMPU

eval echo \${${AANET[eth0]}[1]}
1500

eval echo \${${AANET[eth0]}[3]}
17647

eval echo \${${AANET[eth0]}[7]}
35426

eval echo \${${AANET[eth0]}[@]:3:5}
17647 0 0 0 35426

用于分配临时变量:

eval currentBin=\${${AANET[eth0]}[3]} currentBout=\${${AANET[eth0]}[7]}
echo $currentBout 
35426
echo $currentBin 
17647

甚至:

eval "declare -a currentVals=(\${${AANET[eth0]}[@]:3:8})"
echo ${currentVals[0]}
17647
echo ${currentVals[4]}
35426
echo ${currentVals[@]}
17647 0 0 0 35426 0 0 0

编辑

好的,如果没有 eval!

for aKey in ${!AANET[@]};do
    fields=(${AANET[$aKey]}{[1],[3],[7]});
    echo $aKey ${!fields} ${!fields[1]} ${!fields[2]}
  done |
    xargs printf "%-9s %12s %12s %12s\n" IFace MTU RX TX

IFace              MTU           RX           TX
venet0            1500            0            0
eth1              1500      6400292      6942577
eth0              1500        17647        35426
lo               16436           83           83
于 2012-12-07T13:55:06.790 回答