Перебрать массив строк в Bash?

758

Я хочу написать script, который будет проходить через 15 строк (возможно, массив)? Возможно ли это?

Что-то вроде:

for databaseName in listOfNames
then
  # Do something
end
Теги:
arrays

15 ответов

1234
Лучший ответ

Вы можете использовать его следующим образом:

## 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"
                )
  • 15
    каким-то образом "" около $ i требуется для доступа к каждому элементу массива, вы можете объяснить почему или указать на материал, который я могу прочитать, чтобы понять его? Спасибо!
  • 86
    В BASH безопаснее заключать переменную в кавычки, используя "" для случаев, когда $i может содержать пробелы или расширяемые символы оболочки.
Показать ещё 17 комментариев
434

Это возможно, конечно.

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

См. Bash Для получения подробной информации о циклах, пока и до.

  • 14
    В чем проблема с этим подходом? В простых случаях это работает и, следовательно, более интуитивно понятно, чем ответ @ anubhava.
  • 13
    Это особенно хорошо работает с подстановкой команд, например, for year in $(seq 2000 2013) .
Показать ещё 6 комментариев
91

Ни один из этих ответов не содержит счетчика...

#!/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
  • 6
    Это должен быть принятый ответ, единственный, который работает, когда элементы массива содержат пробелы.
  • 64
    Вы должны начать с 0 для здравомыслия, конечно.
Показать ещё 5 комментариев
76

В том же духе, что и 4-й ответ:

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

В. В именах нет пробелов:

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

Примечания

  • Во втором примере использование 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] может быть указанному в script, чтобы разрешить другие пробелы (т.е. IFS='\n' или для MacOS IFS='\r')
  2. Мне тоже нравится принятый ответ:) - Я включил эти фрагменты в качестве других полезных способов, которые также отвечают на вопрос.
  3. Включение #!/bin/bash в начало файла script указывает среду выполнения.
  4. Мне потребовались месяцы, чтобы понять, как это сделать просто:)

Другие источники (во время цикла чтения)

  • 0
    Это создает впечатление, что eol используется в качестве разделителей строк, и поэтому в строках допускаются пробелы. Однако строки с пробелами далее разделяются на подстроки, что очень и очень плохо. Я думаю, что этот ответ stackoverflow.com/a/23561892/1083704 лучше.
  • 1
    @Val, я добавил код комментария со ссылкой на IFS . (Для всех IFS позволяет указать конкретный разделитель, который позволяет другим пробелам включаться в строки без разделения на подстроки).
Показать ещё 2 комментария
16

Вы можете использовать синтаксис ${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
12

Это также легко читать:

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

#Loop
for Path in "${FilePath[@]}"
do
    echo "$Path"
done
  • 3
    Это понятно и работает для меня (в том числе с пробелами и подстановкой переменных в элементах массива FilePath) только тогда, когда я правильно установил переменную IFS перед определением массива FilePath: IFS=$'\n' Это может работать и для других решений в этот сценарий.
4

Массив объявления не работает для оболочки 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
    попробуйте подсветку кода в редакторе, чтобы ваш код выглядел хорошо.
  • 4
    Полезно знать, но этот вопрос касается bash.
Показать ещё 2 комментария
3

Попробуйте это. Он работает и тестируется.

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 отдельно, во втором он напечатает список файлов вместо *
  • 5
    ... прежде чем вызывать что-то "проверенное" в оболочке, обязательно проверьте угловые случаи, среди которых есть пробелы и глобусы.
2
List=(Item1 Item2 Item3 Item4 Item5 Item6)
for Item in ${List[@]} 
do
echo $Item 
done

OUT PUT:

 Item1
 Item2
 Item3
 Item4
 Item5
 Item6

@

List=(
     Item1 
     Item2
     Item3
     Item4
     Item5
     Item6
     'Item 7 needs to be quoted to preserve spaces'
     )

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

OUT PUT:

Item1
Item2
Item3
Item4
Item5
Item6
Item 7 needs to be quoted to preserve spaces

*

List=(
     Item1 
     Item2
     Item3
     Item4
     Item5
     Item6
     'Item 7 needs to be quoted to preserve spaces'
     )

for Item in "${List[*]}" 
    do
        echo $Item
        echo $'\n' 
    done

OUT PUT с использованием * вместо @производит:

Item1 Item2 Item3 Item4 Item5 Item6 Item 7 needs to be quoted to preserve spaces

.

List=()
for Item in `seq 1 1 16`
    do 
       List+=(Item$Item)
    done
for Item in ${List[*]}
    do 
        echo $Item
    done

Вывод:

Item1
Item2
Item3
Item4
Item5
Item6
Item7
Item8
Item9
Item10
Item11
Item12
Item13
Item14
Item15
Item16
  • 2
    ЭТО НЕ ВЕРНО. Должен быть "${List[@]}" чтобы быть правильным, с кавычками . ${List[@]} не так. ${List[*]} не так. Попробуйте List=( "* first item *" "* second item *" ) - вы получите правильное поведение for item in "${List[@]}"; do echo "$item"; done , но не из любого другого варианта.
  • 0
    Да, специальные символы интерпретируются, что может быть или не быть желательным. Я обновил ответ, чтобы включить.
Показать ещё 21 комментарий
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
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
2

Если вы используете оболочку Korn, существует " set -A databaseName", иначе есть " declare -a databaseName"

Чтобы написать script для всех оболочек,

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

done

Это должно быть работа над всеми оболочками.

  • 1
    Нет, это не так: $ bash --version - версия GNU bash, версия 4.3.33 (0) -релиз (amd64-portbld-freebsd10.0) $ set -A databaseName=("db1" "db2" ....) || declare -a databaseName=("db1" "db2" ....) bash: синтаксическая ошибка рядом с неожиданным токеном `('
1

Это похоже на user2533809 ответ, но каждый файл будет выполнен как отдельная команда.

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

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

Возможная первая строка каждого Bash script/session:

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

Используйте, например:

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

Можно подумать: echo интерпретирует -e как параметр здесь

-2

Я просматриваю массив моих проектов для обновления git pull:

#!/bin/sh
projects="
web
ios
android
"
for project in $projects do
    cd  $HOME/develop/$project && git pull
end
  • 7
    Хотя этот фрагмент кода может решить вопрос, в том числе объяснение действительно помогает улучшить качество вашего сообщения. Помните, что вы отвечаете на вопрос читателей в будущем, и эти люди могут не знать причин, по которым вы предлагаете код. Также постарайтесь не переполнять ваш код пояснительными комментариями, это снижает удобочитаемость кода и пояснений!

Ещё вопросы

Сообщество Overcoder
Наверх
Меню