วนซ้ำสตริงใน Bash หรือไม่


1501

ฉันต้องการเขียนสคริปต์ที่วนรอบ 15 สาย (อาจเป็นไปได้หรือไม่) เป็นไปได้ไหม

สิ่งที่ต้องการ:

for databaseName in listOfNames
then
  # Do something
end

คำตอบ:


2389

คุณสามารถใช้สิ่งนี้:

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

776

แน่นอนว่าเป็นไปได้

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

ดูBash Loops ในขณะและจนกว่าจะมีรายละเอียด


18
ปัญหาเกี่ยวกับวิธีการนี้คืออะไร? ในกรณีง่าย ๆ ดูเหมือนว่าจะใช้งานได้ง่ายกว่าคำตอบของ @ anubhava
ดร. Jan-Philip Gehrcke

17
for year in $(seq 2000 2013)นี้ทำงานได้ดีกับแทนคำสั่งเช่น
แบรดโคช์

23
ปัญหาคือเขาถามเกี่ยวกับการวนซ้ำผ่านอาร์เรย์
mgalgs

20
วิธีการ 'ประกาศ' ทำงานได้ดีที่สุดหากคุณต้องวนซ้ำแถวเดียวกันในสถานที่มากกว่าหนึ่งแห่ง วิธีนี้สะอาด แต่ยืดหยุ่นน้อยกว่า
StampyCode

15
ทำไมไม่ # 1 นี้ DATABASES="a b c d e f"มันทำความสะอาดและคุณสามารถนำมาใช้อาร์เรย์เพียงโดยการตั้งค่าสตริงคือ
Nerdmaster

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

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 และลูปจะไม่ประมวลผลองค์ประกอบทั้งหมด .
Mr 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` 

มันอยู่ถัดจากหมายเลขหนึ่งบนแป้นพิมพ์ของคุณและเหนือปุ่มแท็บ บนแป้นพิมพ์ภาษาอังกฤษแบบอเมริกันมาตรฐาน

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
ฉันยังคงขอแนะนำการแสดงที่ถูกต้องมากขึ้นวิธีการ / ที่แข็งแกร่งแรก ผู้คนมักจะใช้คำตอบแรกที่ดูเหมือนจะใช้ได้ผล หากคำตอบนั้นมีคำเตือนที่ซ่อนอยู่พวกเขาอาจเปิดเผยตัวเองในภายหลังเท่านั้น (มันไม่ได้เป็นเพียงสัญลักษณ์ที่เสียจากการขาด quoting; List=( "first item" "second item" )จะถูกแบ่งออกเป็นfirst, item, second, itemเช่นกัน)
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. ในตัวอย่างที่สองการใช้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 ถูกใช้เป็นตัวคั่นสตริงและดังนั้นอนุญาตให้ใช้ช่องว่างภายใน whitespaces อย่างไรก็ตามสตริงที่มีช่องว่างจะถูกแยกออกเป็นส่วนย่อยซึ่งเป็นสิ่งที่แย่มาก ฉันคิดว่าคำตอบนี้stackoverflow.com/a/23561892/1083704ดีกว่า
Val

1
@Val IFSฉันเพิ่มความคิดเห็นรหัสที่มีการอ้างอิงถึง (สำหรับทุกคนIFSให้หนึ่งระบุตัวคั่นที่เฉพาะเจาะจงซึ่งช่วยให้ช่องว่างอื่น ๆ ที่จะรวมอยู่ในสตริงโดยไม่ต้องแยกออกเป็นสารตั้งต้น)
user2533809

7
มันไม่ได้ผลสำหรับฉัน $databaseNameเพียงแค่มีรายการทั้งหมดจึงทำซ้ำเพียงครั้งเดียว
AlikElzin-kilaka

@ AlikElzin-kilaka คำตอบของฉันด้านล่างแก้ปัญหานี้เพื่อให้วงถูกเรียกใช้สำหรับทุกบรรทัดของสตริง
Jamie

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-loop แบบดั้งเดิม ( for (( i=0; i<${#arr[@]}; i++ )))

( ${!arr[@]}และ$iไม่จำเป็นต้องเสนอราคาเพราะเป็นเพียงตัวเลขบางคนอาจแนะนำให้อ้างถึงพวกเขาต่อไป แต่นั่นเป็นเพียงการตั้งค่าส่วนตัว)


2
นี่ควรเป็นคำตอบที่เลือก 1: เรียบง่ายและอ่านง่าย 2: มันจัดการกับ whitespace อย่างถูกต้องไม่มี IFS ไร้สาระได้รับในทาง 3: มันจัดการ sparse arrays ได้อย่างถูกต้อง
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' สิ่งนี้อาจใช้ได้กับโซลูชันอื่นในสถานการณ์นี้เช่นกัน
Alan Forsyth

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.sh, chmod +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
เป็นการดีที่จะรู้ แต่คำถามนี้เกี่ยวกับทุบตี
แบรดโคช์ส

1
ข้อบกพร่องมากมายที่นี่ ไม่สามารถมีรายการรายการที่มีช่องว่างไม่สามารถมีรายการรายการที่มีตัวอักษรแบบกลม for i in ${foo[*]}โดยทั่วไปมักเป็นสิ่งที่ผิด - for i in "${foo[@]}"เป็นรูปแบบที่รักษาขอบเขตของรายการดั้งเดิมและป้องกันการขยายตัวของ glob และเสียงสะท้อนต้องเป็นecho "$i"
Charles Duffy

4

คล้ายกับคำตอบของผู้ใช้ 2533809 แต่แต่ละไฟล์จะถูกดำเนินการเป็นคำสั่งแยกต่างหาก

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

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

3

หากคุณกำลังใช้ Korn เชลล์จะมี " set -A databaseName " มิฉะนั้นจะมี " ประกาศ -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) - ปล่อย (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=( "*" ); ในกรณีแรกมันจะพิมพ์helloและworldแยกในครั้งที่สองมันจะพิมพ์รายชื่อไฟล์แทน*
Charles Duffy

6
... ก่อนที่จะเรียกบางสิ่งบางอย่าง "ทดสอบ" ในเปลือกให้แน่ใจว่าได้ตรวจสอบกรณีมุมซึ่งช่องว่างและ globs มีทั้งในหมู่
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
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.