使用for循环seq分配动态bash变量名

sto*_*obi 4 linux bash ubuntu

所以我想做点什么,不确定是否可能.我有以下代码:

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')
Run Code Online (Sandbox Code Playgroud)

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

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

mkl*_*nt0 5

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[@]}"
Run Code Online (Sandbox Code Playgroud)

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

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

    • 因此,只需分配元素$i工作,而无需预先确定数组的大小.
  • 注意如何$i需要与前缀$在数组下标,因为数组下标在被评价的算术上下文(在其中相同的上下文$(( ... ))表达式进行求值).


至于你做错什么:

group$i=...
Run Code Online (Sandbox Code Playgroud)

不被Bash识别为变量赋值,因为 - 字面意思 - group$i不是有效的标识符(变量名).

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


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

通过涉及可变声明内建 [命令]如declare,local,或export,将强制击执行扩展第一,扩展group$i它传递给内置之前为有效的变量名.

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

  • 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'
Run Code Online (Sandbox Code Playgroud)

枚举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
Run Code Online (Sandbox Code Playgroud)