Bashで文字列の配列をループしますか?


1502

15個の文字列(配列か?)をループするスクリプトを書きたいのですが、それは可能ですか?

何かのようなもの:

for databaseName in listOfNames
then
  # Do something
end

回答:


2392

次のように使用できます。

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

777

もちろん、それは可能です。

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

詳細については、bashループを参照してください。


18
このアプローチの問題は何ですか?単純なケースではうまくいくようで、@ anubhavaの答えよりも直感的です。
Jan-Philip Gehrcke博士、2012

17
これは、コマンド置換などで特にうまく機能しfor year in $(seq 2000 2013)ます。
Brad Koch

23
問題は、彼が配列を反復することについて尋ねたことです。
mgalgs 2013

20
「宣言」アプローチは、複数の場所で同じ配列を反復処理する必要がある場合に最適に機能します。このアプローチは簡潔ですが、柔軟性が低くなります。
StampyCode 2014年

15
なぜこれが#1ではないのですか?よりクリーンで、文字列を設定するだけで配列を簡単に再利用できますDATABASES="a b c d e f"
Nerdmaster 14

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 14年

1
これは、カウンタを使用した例の出力のためだけのものです。それを変更することも非常に簡単で、まったく同じように機能します。
caktux

7
最後のエコーはバギーです。定数を引用する必要はありません、展開を引用する必要があります、または次のように両方を安全に引用できます:echo "$i / ${arraylength} : ${array[$i-1]}"-それ以外の場合、$iグロブが含まれている場合は展開され、タブが含まれている場合は変更されますスペースなど
Charles Duffy

10
@bzeaman、確かに-しかし、そのようなことについてずさんな場合は、正しいことを証明するためにコンテキスト分析(先ほど行ったように)が必要であり、そのコンテキストが変更された場合、またはコードが別の場所で再利用された場合、またはその他の理由により、予期しない制御フローが発生する可能性があります。堅牢な方法で記述し、コンテキストに関係なく正しいものにします。
Charles Duffy、

5
これはまばらな配列、つまり配列に要素がない場合には機能しません。たとえば、配列要素A [1] = 'xx'、A [4] = 'yy'およびA [9] = 'zz'がある場合、長さは3になり、ループはすべての要素を処理しません。
Ed氏、

145

はい

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

リストへの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番の横、またはタブキーの上にあります。標準的なアメリカ英語のキーボード。

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コマンドラインリファレンスマニュアル:シェルにとっての特定の文字または単語の特別な意味。


7
これは間違っています。する必要があり"${List[@]}"、正しいことが引用符で${List[@]}間違っている。${List[*]}間違っている。試してくださいList=( "* first item *" "* second item *" )-の正しい動作が得られますfor item in "${List[@]}"; do echo "$item"; doneが、他のバリアントからは得られません。
Charles Duffy

はい、特殊文字は解釈されますが、これは望ましい場合と望ましくない場合があります。回答を更新してを含めました。
FireInTheSky

2
それでも、より正確で堅牢なアプローチを最初に示すことを強くお勧めします。人々はしばしばそれがうまくいくように見える最初の答えをとります。その答えに隠された警告がある場合、彼らは後で自分自身を明らかにするかもしれません。(これは、引用の欠如によって破壊されているだけで、ワイルドカードではありません。List=( "first item" "second item" )に分割されますfirstitemseconditem同様)。
Charles Duffy、

また、ベストプラクティスに反して、ls出力を解析する可能性のある例の使用を避けることも検討してください。
Charles Duffy、

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. 2番目の例では、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'、またはMacOSの場合)IFS='\r'
  2. 私も受け入れられた回答が好きです:)-これらのスニペットを、質問にも回答する他の役立つ方法として含めました。
  3. #!/bin/bashスクリプトファイルの先頭に含めると、実行環境を示します。
  4. これを簡単にコーディングする方法を理解するのに数か月かかりました:)

その他のソース(ループの読み取り中


これにより、eolが文字列の区切り文字として使用されているように見えるため、文字列内で空白を使用できます。ただし、空白を含む文字列はさらに部分文字列に分割されるため、非常に悪いです。この答えはstackoverflow.com/a/23561892/1083704の方が良いと思います。
Val

1
@Val、への参照を含むコードコメントを追加しましたIFS。(誰にとっても、IFS特定の区切り文字を指定することができます。これにより、他の空白を部分文字列に分離することなく文字列に含めることができます)。
user2533809 14

7
これは私にはうまくいきません。$databaseNameリスト全体が含まれるだけなので、1回の反復のみが行われます。
AlikElzin-kilaka 2016

@ 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の正解に:ループの基本構文が次の場合:

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シェルでは機能しません。Kornシェルについては、以下の例を使用してください。

promote_sla_chk_lst="cdi xlob"

set -A promote_arry $promote_sla_chk_lst

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

2
エディターでコードハイライターを試して、コードの見栄えを良くしてください。

5
知っておきたいことですが、この質問はbashに関するものです。
Brad Koch

1
Lotsaのバグはこちら。スペースを含むリストエントリを含めることはできません。グロブ文字を含むリストエントリを含めることはできません。for i in ${foo[*]}基本的に常に間違っている- for i in "${foo[@]}"元のリストの境界を保持し、グロブの展開を防ぐフォームです。そして、エコーはする必要がありますecho "$i"
チャールズ・ダフィー

4

これはuser2533809の回答に似ていますが、各ファイルは個別のコマンドとして実行されます。

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

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

3

Kornシェルを使用している場合は、「set -A databaseName」があり、それ以外の場合は「declare -a databaseNameがあります。」があり。

すべてのシェルで機能するスクリプトを作成するには、

 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)-release(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個別に印刷されます。2番目のケースでは、代わりにファイルのリストが印刷されます*
Charles Duffy

6
...シェルで「テスト済み」のものを呼び出す前に、空白とグロブの両方が含まれているコーナーケースを確認してください。
Charles Duffy

2

すべてのBashスクリプト/セッションの可能な最初の行:

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

例:

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

検討してもよい:ここでオプションとしてecho解釈-eする


2

シングルラインループ、

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

次のような出力が得られます。

db_a
db_b
db_c

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 16
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.