名称中带有变量的 bash 数组

名称中带有变量的 bash 数组

感谢您对以下问题的帮助:

我正在尝试设置一个数组,其中包含一个变量作为数组名称的一部分,例如:(Arr_$COUNTER其中$COUNTER根据循环计数进行更改)

我尝试过的每种可能的方法都会出现错误,例如“错误替换”或“意外标记附近的语法错误”

这是整个流程:

  1. 有一个文件包含多行。每行有 6 个值,用空格分隔

    10 20 30 40 50 60  
    100 200 300 400 500 600
    
  2. 该脚本旨在从文件中读取每一行,并将其声明为一个数组(行号是变量。

  3. 作为测试,应该打印每个值,并且最终将对每个值执行另一个函数。

    #!/bin/bash
    COUNTER=1
    LINES=`wc -l VALUES_FILE.txt | awk '{print $1}'`
    echo "Total number of lines "$LINES
    echo
    while [ $COUNTER -le $LINES ]
    do
    echo "Counter value is $COUNTER"
    field=`awk "NR == $COUNTER" VALUES_FILE.txt`
    echo "Field = $field"
    declare -a "arr$COUNTER=($field)"
    echo "arr$COUNTER[0] = ${arr$COUNTER[0]}"
    echo "arr$COUNTER[1] = ${arr$COUNTER[1]}"
    echo "arr$COUNTER[2] = ${arr$COUNTER[2]}"
    echo "arr$COUNTER[3] = ${arr$COUNTER[3]}"
    echo "arr$COUNTER[4] = ${arr$COUNTER[4]}"
    echo "arr$COUNTER[5] = ${arr$COUNTER[5]}"
    let COUNTER=COUNTER+1
    echo
    done
    echo "The End"
    echo
    

结果如下:

总行数 2

计数器值为 1
字段 = 10 20 30 40 50 60
./sort.sh:第 12 行:arr$COUNTER[0] = ${arr$COUNTER[0]}:错误替换
结束

为了使其正常工作,应该更改/修复哪些内容?

感谢 !

答案1

一些想法:

  1. 变量值的“参数扩展”(${...} 部分):

    echo "arr$COUNTER[0] = ${arr$COUNTER[0]}"
    

    不管用。您可以使用 eval 来解决(但我不推荐它):

    eval echo "arr$COUNTER[0] = \${arr$COUNTER[0]}"
    

    该行可以写成这样:

    i="arr$COUNTER[0]"; echo "$i = ${!i}"
    

    这在 Bash 中称为间接(!)。

  2. 此行也出现类似的问题:

    declare -a "arr$COUNTER=($field)"
    

    应该分为两行,并使用 eval:

    declare -a "arr$COUNTER"
    eval arr$COUNTER\=\( \$field \)
    

    再次强调,我不建议使用 eval(在本例中)。

  3. 当您将整个文件读入 shell 的内存时,我们不妨使用更简单的方法将所有行放入数组中:

    readarray -t lines <"VALUES_FILE.txt"
    

    这应该比为每行调用 awk 更快。

包含上述所有内容的脚本可能是:

#!/bin/bash
valfile="VALUES_FILE.txt"

readarray -t lines <"$valfile"             ### read all lines in.

line_count="${#lines[@]}"
echo "Total number of lines $line_count"

for ((l=0;l<$line_count;l++)); do
    echo "Counter value is $l"             ### In which line are we?
    echo "Field = ${lines[l]}"             ### kepth only to help understanding.
    k="arr$l"                              ### Build the variable arr$COUNTER
    IFS=" " read -ra $k <<<"${lines[l]}"   ### Split into an array a line.
    eval max=\${#$k[@]}                    ### How many elements arr$COUNTER has?
    #echo "field $field and k=$k max=$max" ### Un-quote to "see" inside.
    for ((j=0;j<$max;j++)); do             ### for each element in the line.
        i="$k[$j]"; echo "$i = ${!i}"      ### echo it's value.
    done
done
echo "The End"
echo

然而,如果我们能够在 AWK 中执行您需要的内容,AWK 可能会更快。


类似的处理可以在 awk 中完成。假设 6 个值将用作 IP(其中 4 个),另外两个是数字和纪元时间。

只是一个非常简单的 AWK 脚本示例:

#!/bin/sh
valfile="VALUES_FILE.txt"
awk '
NF==6 { printf ( "IP: %s.%s.%s.%s\t",$1,$2,$3,$4)
        printf ( "number: %s\t",$5+2)
        printf ( "epoch: %s\t",$6)
        printf ( "\n" )
    }
' "$valfile"

只需提出一个包含详细信息的新问题即可。

答案2

如果将名称和索引都分配给变量,则可以使用变量间接寻址:

s="arr$COUNTER[0]"
echo "arr$COUNTER[0] = ${!s}"

答案3

在 1 维数组中存储多维数组数据的标准方法是将每一行按偏移量存储到数组中。

元素(i,j)将位于索引处,i*m + j其中i是从零开始的行索引,j是从零开始的列索引,m是列数。

这也使得读取数据变得更容易,因为我们只需获取输入文件,将所有空格更改为换行符并使用readarray.

在命令行上:

$ readarray -t arr < <( tr -s ' ' '\n' <data )
$ printf '%s\n' "${arr[@]}"
10
20
30
40
50
60
100
200
300
400
500
600

我们可以计算出数据中的列数

$ m=$( awk '{ print NF; exit }' <data )

以及行数:

$ n=$( wc -l <data )

然后我们可以像往常一样在双循环中迭代列和行:

for (( i = 0; i < n; ++i )); do
    for (( j = 0; j < m; ++j )); do
        printf '%4d' "${arr[i*m + j]}"
    done
    printf '\n'
done

对于给定的数据,这将生成

  10  20  30  40  50  60
 100 200 300 400 500 600

理想情况下,您应该使用支持多维数组的语言,例如 Perl 或 Python 或 C。也就是说,如果您实际上需要将整组数据存储在内存中并且无法逐行处理它。

对于逐行处理,awk将是替换bash(任何语言将是替换任何类型数据处理的 shell 的良好候选者):

awk '{ for (i = 1; i <= NF; ++i) printf("%4d", $i); printf("\n") }' data

答案4

您可以使用 生成名称eval,例如

eval declare -a '"arr'$COUNTER'=($field)"'

本质上引用了除您想要评估的元字符之外的所有元字符。

所以...如果$COUNTER是 1,你的脚本就可以了

declare -a "arr1=($field)"

进一步阅读:

相关内容