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

时间:2016-02-24 02:41:31

标签: 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')

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

  

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

3 个答案:

答案 0 :(得分: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[@]}"

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

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

    • 因此,只需分配给元素$i就可以工作,而无需预先确定数组的大小。
  • 请注意$i 在数组下标中以$为前缀,因为数组下标是在算术中计算的上下文(评估$(( ... ))表达式的相同上下文)。

至于你做错了什么

group$i=...

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

因为它不是,Bash继续解析,直到找到下一个shell元字符,然后将生成的单词解释为命令来执行,在您的情况下会导致错误消息{{ 1}}。

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

通过涉及变量声明的内置 [命令],例如group0=j: command not founddeclarelocal,您可以强制Bash执行扩展第一个,在将export传递给内置函数之前将group$i扩展为有效的变量名。

  • user2683246's answer通过使用declare(或者,如果需要函数内的局部变量,local)来创建次佳方法变量。

  • Soren's answer使用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

答案 1 :(得分:4)

使用declare group$i=...代替group$i=...

答案 2 :(得分:1)

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

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

with declare

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

其中export使值可用于子流程,而declare仅在同一脚本中可用。