在 Bash 中循环访问字符串数组?

Loop through an array of strings in Bash?

提问人:Mo. 提问时间:1/16/2012 最后编辑:codeforesterMo. 更新时间:1/17/2022 访问量:1927232

问:

我想编写一个遍历 15 个字符串的脚本(可能是数组?这可能吗?

像这样:

for databaseName in listOfNames
then
  # Do something
end
数组 bash shell

评论


答:

1105赞 4ndrew 1/16/2012 #1

当然,这是可能的。

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

有关详细信息,请参阅 Bash Loops for、while 和 until

评论

28赞 Dr. Jan-Philip Gehrcke 8/30/2012
这种方法有什么问题?在简单的情况下,它似乎有效,然后比@anubhava的答案更直观。
21赞 Brad Koch 5/20/2013
这在命令替换时特别有效,例如 .for year in $(seq 2000 2013)
27赞 StampyCode 1/3/2014
如果必须在多个位置遍历同一数组,则“声明”方法效果最佳。这种方法更简洁,但不太灵活。
21赞 Nerdmaster 7/3/2014
为什么不是#1?它更简洁,您只需设置一个字符串即可轻松重用数组,即 .DATABASES="a b c d e f"
17赞 pabouk - Ukraine stay strong 11/24/2014
@Nerdmaster:是的,这是一个更简单(和可移植)的代码,但是在变量(实际上是单个字符串)中存储由空格分隔的字符串列表可以防止单个字符串包含空格。user2533809 的回复解决了这个问题。------ 但还有另一个问题。如果列表很长并且必须更新,则此类更新操作需要每次都复制完整列表,因此它们将非常慢。
3384赞 anubhava 1/16/2012 #2

你可以像这样使用它:

## 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"
                )

评论

161赞 Gabriel Staples 6/15/2020
请注意,“${arr[@]}”周围的双引号非常重要。如果没有它们,for 循环将按子字符串分解数组,子字符串由字符串中的任何空格分隔,而不是数组中的整个字符串元素分隔。即:如果你有 ,那么会错误地迭代 6 次,因为每个字符串变成 2 个子字符串,由字符串中的空格分隔,而会根据需要正确迭代 3 次,尽管其中有一个空格,但将每个字符串保持为一个单元。declare -a arr=("element 1" "element 2" "element 3")for i in ${arr[@]}for i in "${arr[@]}"
7赞 Bhanu 11/3/2012 #3

declare 数组不适用于 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赞 dove 11/3/2012
尝试在编辑器中使用代码荧光笔,使代码看起来不错。
5赞 Brad Koch 5/20/2013
很高兴知道,但这个问题是关于 bash 的。
1赞 Charles Duffy 7/9/2016
Lotsa 错误在这里。不能有带空格的列表条目,不能有带通球字符的列表条目。 基本上总是错误的事情 -- 是保留原始列表边界并防止 glob 扩展的形式。并且回声需要for i in ${foo[*]}for i in "${foo[@]}"echo "$i"
8赞 SK. 1/17/2014 #4

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

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

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

评论

3赞 Charles Duffy 7/9/2016
这实际上并不能正常工作。尝试 或 ;在第一种情况下,它将单独打印,在第二种情况下,它将打印文件列表,而不是array=( "hello world" )arrray=( "*" )helloworld*
7赞 Charles Duffy 7/9/2016
...在 shell 中调用“测试”的内容之前,请务必检查极端情况,其中空格和 glob 都在其中。
116赞 dajon 1/17/2014 #5

本着与 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] 分隔符以允许其他空格(即 ,或 对于 MacOSIFS='\n'IFS='\r')
  2. 我也喜欢公认的答案:) - 我已经将这些片段作为其他有用的方法,也可以回答这个问题。
  3. 包含在脚本文件的顶部表示执行环境。#!/bin/bash
  4. 我花了几个月的时间才弄清楚如何简单地编写这个代码:)

其他来源 (while 读取循环)

评论

0赞 Val 7/11/2014
这给人的印象是 eol 被用作字符串分隔符,因此字符串中允许有空格。但是,带有空格的字符串会进一步分隔为子字符串,这是非常非常糟糕的。我认为这个答案 stackoverflow.com/a/23561892/1083704 更好。
1赞 dajon 7/22/2014
@Val,我添加了代码注释,并引用了 .(对于每个人,让我们指定一个特定的分隔符,它允许其他空格包含在字符串中,而不被分隔为子字符串)。IFSIFS
12赞 AlikElzin-kilaka 7/18/2016
这对我不起作用。 只包含整个列表,因此只执行一次迭代。$databaseName
0赞 Jamie 10/17/2016
@AlikElzin-基拉卡 我下面的答案解决了这个问题,以便为字符串的每一行运行循环。
56赞 Fizer Khan 2/17/2014 #6

您可以使用${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
363赞 caktux 3/16/2014 #7

这些答案中没有一个包括计数器......

#!/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=0; i<${arraylength}; i++ ));
do
  echo "index: $i, value: ${array[$i]}"
done

输出:

index: 0, value: one
index: 1, value: two
index: 2, value: three

评论

2赞 caktux 7/9/2015
这只是为了示例的输出与计数器。改变这一点也是相当微不足道的,而且工作原理是一样的。
7赞 Charles Duffy 7/9/2016
最后的回声是有问题的。你不需要引号常量,你需要引号扩展,或者可以安全地只引号,如下所示: -- 否则,如果你包含一个球,它将被展开,如果它包含一个制表符,它将被更改为一个空格,等等。echo "$i / ${arraylength} : ${array[$i-1]}"$i
11赞 Charles Duffy 9/2/2016
@bzeaman,当然 -- 但是如果你对这些事情很草率,那么它需要上下文分析(就像你刚才所做的那样)来证明某些东西是正确的,如果上下文发生变化或代码在不同的地方重用,或者由于任何其他原因,可能会出现意外的控制流,则需要重新分析。以稳健的方式编写它,无论上下文如何,它都是正确的。
11赞 Mr Ed 12/12/2017
这不适用于稀疏数组,即如果数组缺少元素。例如,如果您有数组元素 A[1]='xx'、A[4]='yy' 和 A[9]='zz',则长度将为 3,并且循环不会处理所有元素。
2赞 mgutt 11/28/2021
无需获取数组的长度。简单易用for i in "${!array[@]}";
4赞 The smart Eagle 9/25/2014 #8

如果您使用的是 Korn shell,则有“set -A databaseName”,否则有“declare -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

它应该适用于所有外壳。

评论

4赞 Bernie Reiter 1/18/2017
不,它没有: GNU bash,版本 4.3.33(0)-release (amd64-portbld-freebsd10.0) bash:意外标记“(”附近的语法错误$ bash --version$ set -A databaseName=("db1" "db2" ....) || declare -a databaseName=("db1" "db2" ....)
0赞 Jacksonkr 9/15/2023
@BernieReiter确保你使用的是 而不是$ ./myscript.sh$ sh myscript.shchmod +x myscript.sh
28赞 Treethawat Thanawachiramate 7/3/2015 #9

这也很容易阅读:

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

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

评论

5赞 Alan Forsyth 1/22/2016
只有当我在 FilePath 数组定义之前正确设置 IFS 变量时,这个才清楚并且对我有用(包括 FilePath 数组元素中的空格和变量替换): 在这种情况下,这可能也适用于其他解决方案。IFS=$'\n'
3赞 elf12 9/13/2015 #10

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

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

例如,使用:

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

可以考虑:此处解释为选项echo-e

13赞 nroose 3/20/2016 #11
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
6赞 Jamie 10/17/2016 #12

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

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

while read -r line; do
    echo line: "$line"
done <<< "$names"
2赞 jiahut 12/8/2016 #13

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

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

评论

7赞 kayess 12/8/2016
虽然这个代码片段可能会解决这个问题,但包括一个解释确实有助于提高你的帖子的质量。请记住,您正在为将来的读者回答问题,而这些人可能不知道您的代码建议的原因。也请尽量不要用解释性注释来填充你的代码,这会降低代码和解释的可读性!
0赞 Akhil 8/21/2020
您应该在脚本中解释变量行为IFS
3赞 Mohideen bin Mohammed 6/27/2017 #14

单线循环,

 declare -a listOfNames=('db_a' 'db_b' 'db_c')
 for databaseName in ${listOfNames[@]}; do echo $databaseName; done;

你会得到这样的输出,

db_a
db_b
db_c
286赞 abc 9/12/2017 #15

是的

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[*]}

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

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

CSV 变量或文件添加到列表中
将内部字段分隔符从空格更改为您想要的任何内容。
在下面的示例中,它被更改为逗号

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

如果需要对它们进行编号:

` 

这称为反蜱。将命令放在反勾号内。

`command` 

它位于键盘上的数字 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

更加熟悉 bashes 行为:

在文件中创建列表

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 的特殊含义。

评论

11赞 Charles Duffy 5/5/2018
这是错误的。需要正确,带引号。 是错误的。 是错误的。尝试 -- 您将获得 的正确行为,但不会从任何其他变体中获得正确的行为。"${List[@]}"${List[@]}${List[*]}List=( "* first item *" "* second item *" )for item in "${List[@]}"; do echo "$item"; done
1赞 abc 5/27/2018
是的,特殊字符确实会被解释,这可能是可取的,也可能是不可取的。我更新了答案以包括.
2赞 Charles Duffy 5/27/2018
我仍然强烈建议首先展示更正确/更强大的方法。人们经常接受第一个看起来会起作用的答案;如果这个答案有隐藏的警告,他们可能只会在以后暴露自己。(不仅仅是通配符因缺乏引号而被破坏; 也会被分解成 、 、 和)。List=( "first item" "second item" )firstitemseconditem
1赞 Charles Duffy 5/27/2018
您还可以考虑避免使用可能导致人们解析输出的示例,这违反了最佳实践ls
2赞 Charles Duffy 6/1/2018
你反驳了我从未提出过的说法。我对 bash 的引用语义非常熟悉——参见 stackoverflow.com/tags/bash/topusers
11赞 F. Hauri - Give Up GitHub 12/5/2017 #16

脚本或函数的隐式数组:

除了 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'.
56赞 tckmn 8/11/2018 #17

令人惊讶的是,还没有人发布这个 - 如果你在循环数组时需要元素的索引,你可以这样做:

arr=(foo bar baz)

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

输出:

0 foo
1 bar
2 baz

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

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

评论

5赞 mrm 12/5/2019
这确实应该是选择的答案。1:它简单易读。2:它正确处理空格,没有IFS废话。3:它正确处理稀疏数组。
0赞 mirekphd 7/23/2022
@tckmn:感叹号(使用两次)是必要的吗?似乎错了......
13赞 rashedcs 9/2/2018 #18

简单的方法:

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}
4赞 Mahdi-Malv 10/21/2019 #19

我真正需要的是这样的:

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

例如:

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

(会杀死所有名称中带有 vlc 的进程)

3赞 Dan Bray 2/5/2021 #20

如何遍历数组取决于是否存在换行符。使用换行符分隔数组元素,数组可以称为 ,否则应称为 。以下脚本将明确说明:"$array""${array[@]}"

#!/bin/bash

mkdir temp
mkdir temp/aaa
mkdir temp/bbb
mkdir temp/ccc
array=$(ls temp)
array1=(aaa bbb ccc)
array2=$(echo -e "aaa\nbbb\nccc")

echo '$array'
echo "$array"
echo
for dirname in "$array"; do
    echo "$dirname"
done
echo
for dirname in "${array[@]}"; do
    echo "$dirname"
done
echo
echo '$array1'
echo "$array1"
echo
for dirname in "$array1"; do
    echo "$dirname"
done
echo
for dirname in "${array1[@]}"; do
    echo "$dirname"
done
echo
echo '$array2'
echo "$array2"
echo
for dirname in "$array2"; do
    echo "$dirname"
done
echo
for dirname in "${array2[@]}"; do
    echo "$dirname"
done
rmdir temp/aaa
rmdir temp/bbb
rmdir temp/ccc
rmdir temp
33赞 Kofi 1/17/2022 #21

我在 GitHub 更新中使用了这种方法,我发现它很简单。

## declare an array variable
arr_variable=("kofi" "kwame" "Ama")

## now loop through the above array
for i in "${arr_variable[@]}"
do
   echo "$i"


done
   

您可以使用具有三表达式(C 样式)的计数器遍历 bash 数组值,以读取循环语法的所有值和索引:

declare -a kofi=("kofi" "kwame" "Ama")
 
# get the length of the array
length=${#kofi[@]}

for (( j=0; j<${length}; j++ ));
do
  print (f "Current index %d with value %s\n" $j "${kofi[$j]}")
done

评论

1赞 Fravadona 1/18/2022
bash 数组中可能有孔;通过索引遍历它的正确方法是(也适用于关联数组):for i in "${!arr_variable[@]}"; do printf '%d: %s\n' "$i" "${arr_variable[i]}"; done