使用for循环seq分配dynamicbashvariables名称

所以我想做点什么,不知道有没有可能。 我有以下代码:

for i in {0..5}; do if [[ -f ./user$i ]]; then group$i=$(grep -w "group" ./user0|awk '{print $2}'|perl -lape 's/\s+//sg') 

我想要做的是为每个variables名称的{0..5}所以group1 group2 group3 group4的每个实例分配一个唯一的variables。 然后我会将./user0更改为./user$i,并根据我的顺序创build一个dynamicvariables列表。 这可能吗? 尝试执行此操作时出现以下错误,我不确定我实际上做了哪些bash不喜欢的操作。

test.sh:第16行:group0 = j:找不到命令

Kurt Stutsman在这个问题的评论中提供了正确的指针: 使用Bash 数组来解决你的问题。

这是一个简单的例子:

 groups=() # declare an empty array; same as: declare -a groups for i in {0..5}; do groups[i]="group $i" # dynamically create element with index $i done # Print the resulting array's elements. printf '%s\n' "${groups[@]}" 

有关列举数组${groups[@]}的元素的其他方法,请参阅此答案的底部。

  • bash数组可以动态扩展(甚至可以是稀疏的 – 元素索引不必是连续的)

    • 因此,简单地分配元素$i工作,没有事先大小的数组。
  • 请注意,数组下标中$i不需要以$作为前缀,因为数组下标是在算术上下文中计算的$(( ... ))表达式在同一个上下文中计算)。


至于你做错什么

 group$i=... 

不会被Bash识别为变量赋值,因为 – 从字面上看group$i不是有效的标识符(变量名称)。

因为它不是,Bash继续解析,直到找到下一个shell元字符,然后将结果字解释为要执行的命令 ,在您的情况下会导致错误消息group0=j: command not found


如果出于某种原因,您不想使用数组完全避免此问题,则可以解决此问题

通过涉及declarelocalexport等变量声明的内建命令,您可以强制Bash首先执行扩展,这会在将group$i传递给内建之前将其扩展为有效的变量名称。

  • user2683246的答案通过使用declare (或者,如果函数内部的局部变量是需要的, local )来演示下一个最好的方法来创建变量。

  • Soren的答案使用export ,但只有在创建子进程可见的 环境变量而不是仅仅是shell变量的情况下才是可取的。

注意 :使用这种技术时, 一定要加上RHS的双引号以获取完整的值。 为了显示:

  i=0; declare v$i=$(echo 'hi, there'); echo "$v0" # !! WRONG -> 'hi,': only UP TO 1ST SPACE i=0; declare v$i="$(echo 'hi, there')"; echo "$v0" # OK -> 'hi, there' 

枚举上面创建的groups数组的其他方法:

 # Enumerate array elements directly. for element in "${groups[@]}"; do echo "$element" done # Enumerate array elements by index. for (( i = 0; i < ${#groups[@]}; i++ )); do echo "#$i: ${groups[i]}" done 

使用declare group$i=...而不是仅仅group$i=...

尝试使用这样的导出或声明函数

 for i in {0..5}; do if [[ -f ./user$i ]]; then export group$i=$(grep -w "group" ...... 

与申报

 for i in {0..5}; do if [[ -f ./user$i ]]; then declare group$i=$(grep -w "group" ...... 

在哪里export使子进程可用的值,并且在相同的脚本中declare