在Bash中循环遍历字符串数组?


1499

我想编写一个循环15个字符串(可能是数组)的脚本吗?

就像是:

for databaseName in listOfNames
then
  # Do something
end

Answers:


2383

您可以像这样使用它:

## declare an array variable
declare -a arr=("element1" "element2" "element3")

## now loop through the above array
for i in "${arr[@]}"
do
   echo "$i"
   # or do whatever with individual element of the array
done

# You can access them using echo "${arr[0]}", "${arr[1]}" also

也适用于多行数组声明

declare -a arr=("element1" 
                "element2" "element3"
                "element4"
                )

775

当然可以。

for databaseName in a b c d e f; do
  # do something like: echo $databaseName
done 

有关详细信息请参见Bash循环


18
这种方法有什么问题?在简单的情况下,它似乎可以工作,并且比@anubhava的答案更直观。
Jan-Philip Gehrcke博士

17
这对于命令替换(例如)特别有效for year in $(seq 2000 2013)
布拉德·科赫

22
问题是他问有关遍历数组的问题。
mgalgs 2013年

20
如果您必须在多个位置迭代同一数组,则“声明”方法最有效。这种方法比较干净,但灵活性较差。
StampyCode 2014年

15
为什么不是#1?它更干净,您只需设置一个字符串即可轻松重用该数组DATABASES="a b c d e f"
Nerdmaster 2014年

201

这些答案都没有包含计数器...

#!/bin/bash
## declare an array variable
declare -a array=("one" "two" "three")

# get length of an array
arraylength=${#array[@]}

# use for loop to read all values and indexes
for (( i=1; i<${arraylength}+1; i++ ));
do
  echo $i " / " ${arraylength} " : " ${array[$i-1]}
done

输出:

1  /  3  :  one
2  /  3  :  two
3  /  3  :  three

7
这应该是公认的答案,是当数组元素包含空格时唯一可行的答案。
jesjimher 2014年

1
仅仅是为了示例的输出加上计数器。更改它也很简单,并且工作原理相同。
caktux

7
最后的回声是越野车。您不需要引用常量,您不需要引用扩展,或者可以安全地引用两个,如下所示:echo "$i / ${arraylength} : ${array[$i-1]}"-否则,如果您$i包含一个glob,它将被扩展,如果它包含一个制表符,它将被更改为一个空间等
查尔斯·达菲

10
@bzeaman,当然-但是,如果您对此事草率,则需要进行上下文分析(就像您刚才所做的那样),以证明某些内容正确无误,如果上下文发生更改或代码在其他地方重复使用或用于其他用途,则需要重新分析出于其他原因,可能会出现意外的控制流。以健壮的方式编写它,无论上下文如何,它都是正确的。
查尔斯·达菲

5
这不适用于稀疏数组,即,如果数组缺少元素。例如,如果您有数组元素A [1] ='xx',A [4] ='yy'和A [9] ='zz',则长度为3,循环不会处理所有元素。
艾德先生

144

for Item in Item1 Item2 Item3 Item4 ;
  do
    echo $Item
  done

输出:

Item1
Item2
Item3
Item4

保留空间;单引号或双引号列表条目以及双引号列表扩展。

for Item in 'Item 1' 'Item 2' 'Item 3' 'Item 4' ;
  do
    echo "$Item"
  done

输出:

Item 1
Item 2
Item 3
Item 4

列出多行

for Item in Item1 \
            Item2 \
            Item3 \
            Item4
  do
    echo $Item
  done

输出:

Item1
Item2
Item3
Item4


简单列表变量

List=( Item1 Item2 Item3 )

要么

List=(
      Item1 
      Item2 
      Item3
     )

显示列表变量:

echo ${List[*]}

输出:

Item1 Item2 Item3

遍历列表:

for Item in ${List[*]} 
  do
    echo $Item 
  done

输出:

Item1
Item2
Item3

创建一个要遍历列表的函数:

Loop(){
  for item in ${*} ; 
    do 
      echo ${item} 
    done
}
Loop ${List[*]}

使用clarify关键字(命令)创建列表,该列表在技术上称为数组:

declare -a List=(
                 "element 1" 
                 "element 2" 
                 "element 3"
                )
for entry in "${List[@]}"
   do
     echo "$entry"
   done

输出:

element 1
element 2
element 3

创建一个关联数组。一本字典:

declare -A continent

continent[Vietnam]=Asia
continent[France]=Europe
continent[Argentina]=America

for item in "${!continent[@]}"; 
  do
    printf "$item is in ${continent[$item]} \n"
  done

输出:

 Argentina is in America
 Vietnam is in Asia
 France is in Europe

将CVS变量或文件放入列表中
将内部字段分隔符从空间更改为所需的内容。
在下面的示例中,将其更改为逗号

List="Item 1,Item 2,Item 3"
Backup_of_internal_field_separator=$IFS
IFS=,
for item in $List; 
  do
    echo $item
  done
IFS=$Backup_of_internal_field_separator

输出:

Item 1
Item 2
Item 3

如果需要编号:

` 

这称为反向刻度。将命令放回去号内。

`commend` 

它在键盘上的数字1旁边或在Tab键上方。在标准的美式英语键盘上。

List=()
Start_count=0
Step_count=0.1
Stop_count=1
for Item in `seq $Start_count $Step_count $Stop_count`
    do 
       List+=(Item_$Item)
    done
for Item in ${List[*]}
    do 
        echo $Item
    done

输出为:

Item_0.0
Item_0.1
Item_0.2
Item_0.3
Item_0.4
Item_0.5
Item_0.6
Item_0.7
Item_0.8
Item_0.9
Item_1.0

变得更加熟悉休克行为:

在文件中创建列表

cat <<EOF> List_entries.txt
Item1
Item 2 
'Item 3'
"Item 4"
Item 7 : *
"Item 6 : * "
"Item 6 : *"
Item 8 : $PWD
'Item 8 : $PWD'
"Item 9 : $PWD"
EOF

将列表文件读入列表并显示

List=$(cat List_entries.txt)
echo $List
echo '$List'
echo "$List"
echo ${List[*]}
echo '${List[*]}'
echo "${List[*]}"
echo ${List[@]}
echo '${List[@]}'
echo "${List[@]}"

BASH命令行参考手册:某些字符或单词对Shell的特殊含义。


7
这是错误的。需要"${List[@]}"正确,用引号引起来${List[@]}是错的。${List[*]}是错的。尝试List=( "* first item *" "* second item *" )-您将获得的正确行为for item in "${List[@]}"; do echo "$item"; done,但不会从其他任何变体中获得。
查尔斯·达菲

是的,特殊字符确实会被解释,这可能是也可能不是理想的。我将答案更新为。
FireInTheSky

2
我仍然强烈建议出更正确的/可靠的方法第一。人们通常会选择第一个看起来可行的答案。如果该答案有隐藏的警告,则他们可能只会在以后暴露自己。(这不只是通配符由缺乏引用的折断; List=( "first item" "second item" )将被分解成firstitemseconditem以及)。
查尔斯·达菲

您可能还会考虑避免使用可能导致人们解析ls输出的示例,而这与最佳实践相违背
查尔斯·达菲

1
您驳斥了我从未提出的主张。我对bash的引用语义非常熟悉-请参阅stackoverflow.com/tags/bash/topusers
Charles Duffy

108

本着与4ndrew的回答相同的精神:

listOfNames="RA
RB
R C
RD"

# To allow for other whitespace in the string:
# 1. add double quotes around the list variable, or
# 2. see the IFS note (under 'Side Notes')

for databaseName in "$listOfNames"   #  <-- Note: Added "" quotes.
do
  echo "$databaseName"  # (i.e. do action / processing of $databaseName here...)
done

# Outputs
# RA
# RB
# R C
# RD

B.名称中不能有空格:

listOfNames="RA
RB
R C
RD"

for databaseName in $listOfNames  # Note: No quotes
do
  echo "$databaseName"  # (i.e. do action / processing of $databaseName here...)
done

# Outputs
# RA
# RB
# R
# C
# RD

笔记

  1. 在第二个示例中,using listOfNames="RA RB R C RD"具有相同的输出。

引入数据的其他方式包括:

从stdin读取

# line delimited (each databaseName is stored on a line)
while read databaseName
do
  echo "$databaseName"  # i.e. do action / processing of $databaseName here...
done # <<< or_another_input_method_here
  1. 可以在脚本中指定bash IFS “行分隔符” [ 1 ]分隔符,以允许其他空格(例如IFS='\n',对于MacOSIFS='\r'
  2. 我也喜欢被接受的答案:)-我将这些摘要作为其他有用的方式也可以回答问题。
  3. #!/bin/bash脚本文件的顶部包括指示执行环境。
  4. 我花了几个月的时间弄清楚如何简单地编写此代码:)

其他来源(读取循环时


这会给人以eol用作字符串分隔符的印象,因此在字符串中允许使用空格。但是,带有空格的字符串会进一步分成子字符串,这是非常非常糟糕的。我认为这个答案stackoverflow.com/a/23561892/1083704更好。
Val

1
@Val,我添加了对的引用的代码注释IFS。(对于每个人,IFS让我们指定一个特定的定界符,该定界符允许将其他空格包含在字符串中,而不必分成​​子字符串)。
user2533809

7
这对我不起作用。$databaseName仅包含整个列表,因此仅执行一次迭代。
AlikElzin-kilaka

@ AlikElzin-kilaka我的下面的答案解决了这个问题,以便对字符串的每一行都运行循环。
杰米

42

您可以使用以下语法 ${arrayName[@]}

#!/bin/bash
# declare an array called files, that contains 3 values
files=( "/etc/passwd" "/etc/group" "/etc/hosts" )
for i in "${files[@]}"
do
    echo "$i"
done

31

令人惊讶的是,还没有人发布此内容-如果在遍历数组时需要元素的索引,则可以执行以下操作:

arr=(foo bar baz)

for i in ${!arr[@]}
do
    echo $i "${arr[i]}"
done

输出:

0 foo
1 bar
2 baz

我发现这比“传统” for循环样式(for (( i=0; i<${#arr[@]}; i++ )))优雅得多。

${!arr[@]}并且$i不需要被引用,因为它们只是数字;有些人建议无论如何都要引用它们,但这只是个人喜好。)


2
这确实应该是选择的答案。1:简单易读。2:它正确处理空格,没有IFS废话。3:它可以正确处理稀疏数组。
mrm

20

这也很容易理解:

FilePath=(
    "/tmp/path1/"    #FilePath[0]
    "/tmp/path2/"    #FilePath[1]
)

#Loop
for Path in "${FilePath[@]}"
do
    echo "$Path"
done

4
仅当我在FilePath数组定义之前正确设置IFS变量时,这一点才清楚并且对我有用(包括FilePath数组元素中的空格和变量替换): IFS=$'\n' 在这种情况下,这也可能适用于其他解决方案。
艾伦·福赛斯

8

脚本或函数的隐式数组:

除了anubhava的正确答案之外:如果loop的基本语法是:

for var in "${arr[@]}" ;do ...$var... ;done

有一个特别的情况

当运行脚本或函数,参数在命令行通过将被分配给$@数组变量,就可以访问通过$1$2$3,等等。

可以通过以下方式填充(用于测试)

set -- arg1 arg2 arg3 ...

一个循环这个阵列可以简单地写成:

for item ;do
    echo "This is item: $item."
  done

请注意in不存在保留的工作,也没有数组名称!

样品:

set -- arg1 arg2 arg3 ...
for item ;do
    echo "This is item: $item."
  done
This is item: arg1.
This is item: arg2.
This is item: arg3.
This is item: ....

请注意,这与

for item in "$@";do
    echo "This is item: $item."
  done

然后放入脚本

#!/bin/bash

for item ;do
    printf "Doing something with '%s'.\n" "$item"
  done

将其保存在脚本myscript.shchmod +x myscript.sh,然后

./myscript.sh arg1 arg2 arg3 ...
Doing something with 'arg1'.
Doing something with 'arg2'.
Doing something with 'arg3'.
Doing something with '...'.

功能相同:

myfunc() { for item;do cat <<<"Working about '$item'."; done ; }

然后

myfunc item1 tiem2 time3
Working about 'item1'.
Working about 'tiem2'.
Working about 'time3'.

7
listOfNames="db_one db_two db_three"
for databaseName in $listOfNames
do
  echo $databaseName
done

要不就

for databaseName in db_one db_two db_three
do
  echo $databaseName
done

7

简单方法:

arr=("sharlock"  "bomkesh"  "feluda" )  ##declare array

len=${#arr[*]}  # it returns the array length

#iterate with while loop
i=0
while [ $i -lt $len ]
do
    echo ${arr[$i]}
    i=$((i+1))
done


#iterate with for loop
for i in $arr
do
  echo $i
done

#iterate with splice
 echo ${arr[@]:0:3}

6

该声明数组不适用于Korn shell。将以下示例用于Korn shell:

promote_sla_chk_lst="cdi xlob"

set -A promote_arry $promote_sla_chk_lst

for i in ${promote_arry[*]};
    do
            echo $i
    done

2
尝试在编辑器中使用代码突出显示功能,使您的代码看起来不错。
2012年

5
很高兴知道,但是这个问题是关于bash的。
布莱德·科赫

1
Lotsa的bug在这里。列表项不能带有空格,列表项不能带有glob字符。for i in ${foo[*]}基本上总是错误的事情- for i in "${foo[@]}"是保留原始列表边界并防止全局扩展的形式。回声必须是echo "$i"
Charles Duffy

4

这类似于user2533809的答案,但是每个文件将作为单独的命令执行。

#!/bin/bash
names="RA
RB
R C
RD"

while read -r line; do
    echo line: "$line"
done <<< "$names"

3

如果使用的是Korn Shell,则存在“ set -A databaseName ”,否则存在“ declaration -a databaseName

要编写适用于所有shell的脚本,

 set -A databaseName=("db1" "db2" ....) ||
        declare -a databaseName=("db1" "db2" ....)
# now loop 
for dbname in "${arr[@]}"
do
   echo "$dbname"  # or whatever

done

它应该适用于所有外壳。


3
不,不是这样的: $ bash --versionGNU bash,版本4.3.33(0)-发行版(amd64-portbld-freebsd10.0)$ set -A databaseName=("db1" "db2" ....) || declare -a databaseName=("db1" "db2" ....)bash:意外标记'('
Bernie Reiter

2

尝试这个。它正在工作并经过测试。

for k in "${array[@]}"
do
    echo $k
done

# For accessing with the echo command: echo ${array[0]}, ${array[1]}

3
这实际上不能正常工作。尝试array=( "hello world" ),或arrray=( "*" ); 在第一种情况下,将打印helloworld分开,在所述第二,将打印的文件的列表,而不是的*
查尔斯达菲

6
...在外壳程序中调用“经过测试”的内容之前,请务必检查拐角处的情况,其中包括空格和glob。
查尔斯·达菲

2

每个Bash脚本/会话可能的第一行:

say() { for line in "${@}" ; do printf "%s\n" "${line}" ; done ; }

使用例如:

$ aa=( 7 -4 -e ) ; say "${aa[@]}"
7
-4
-e

可以考虑:这里echo解释-e为选项



2

我真正需要的是这样的东西:

for i in $(the_array); do something; done

例如:

for i in $(ps -aux | grep vlc  | awk '{ print $2 }'); do kill -9 $i; done

(将杀死名称为vlc的所有进程)


1

我遍历我的项目数组以进行git pull更新:

#!/bin/sh
projects="
web
ios
android
"
for project in $projects do
    cd  $HOME/develop/$project && git pull
end

7
尽管此代码段可以解决问题,但提供说明确实有助于提高您的帖子质量。请记住,您将来会为读者回答这个问题,而这些人可能不知道您提出代码建议的原因。也请尽量不要在代码中加入解释性注释,这会降低代码和解释的可读性!
kayess
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.