Цикл через массив строк в Bash?
Я хочу написать сценарий, который проходит через 15 строк (возможно, массив? Это возможно?
что-то типа:
for databaseName in listOfNames
then
# Do something
end
19 ответов
Вы можете использовать его как это:
## 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"
)
это возможно, конечно.
for databaseName in a b c d e f; do
# do something like: echo $databaseName
done
посмотреть Bash петли Для, в то время как и до для сведения.
ни один из этих ответов не включают счетчик...
#!/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
в том же духе, что и ответ 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
Б. Нет пробелов в именах:
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 (перечислены ниже),
- переменные,
- массив (принято отвечать),
- файл...
читать из 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] разделитель может быть указан в скрипте, чтобы разрешить другие пробелы (т. е.
IFS='\n'
, или для MacOSIFS='\r'
) - мне также нравится принятый ответ:) -- я включил эти фрагменты в качестве других полезных способов, которые также отвечают на вопрос.
- в том числе
#!/bin/bash
в верхней части файла сценария указывает среде выполнения. - мне потребовались месяцы, чтобы понять, как это просто закодировать:)
Других Источников (пока читал цикл)
Вы можете использовать синтаксис ${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
да
for Item in Item1 Item2 Item3 Item4 ;
do
echo $Item
done
выход :
Item1
Item2
Item3
Item4
на несколько строк
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[*]}
для сохранения пробелов; одиночные или двойные записи списка кавычек и расширения списка двойных кавычек:
List=(' Item 1 '
' Item 2'
' Item 3'
)
for item in "${List[@]}";
do
echo "$item"
done
выход :
Item 1
Item 2
Item 3
используя ключевое слово 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`
это рядом с номером один на клавиатуре и или выше клавиши 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
Out put is:
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[@]}"
Это также легко читать:
FilePath=(
"/tmp/path1/" #FilePath[0]
"/tmp/path2/" #FilePath[1]
)
#Loop
for Path in "${FilePath[@]}"
do
echo "$Path"
done
массив declare не работает для оболочки 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
неявный массив для скрипта или функции:
кроме Анубхаваправильный ответ: если основной синтаксис для цикла:
for var in "${arr[@]}" ;do ...$var... ;done
есть специальные в случае Баш:
при запуске скрипта или функции, аргументы пройденные в командной строке будут назначены $@
переменной массива, вы можете получить доступ к ,
,
и так далее.
этот может быть заполнен (для теста) по
set -- arg1 arg2 arg3 ...
A цикл над этой массив может быть написано просто:
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'.
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
Если вы используете оболочку 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
это должна быть работа на всех оболочках.
Это похоже на ответ user2533809, но каждый файл будет выполняться как отдельная команда.
#!/bin/bash
names="RA
RB
R C
RD"
while read -r line; do
echo line: "$line"
done <<< "$names"
возможная первая строка каждого сценария/сеанса Bash:
say() { for line in "${@}" ; do printf "%s\n" "${line}" ; done ; }
, например:
$ aa=( 7 -4 -e ) ; say "${aa[@]}"
7
-4
-e
можете считать: echo
интерпретирует -e
как вариант
однострочный цикл,
declare -a listOfNames=('db_a' 'db_b' 'db_c')
for databaseName in ${listOfNames[@]}; do echo $databaseName; done;
вы получите такой результат,
db_a
db_b
db_c
удивлен, что никто еще не опубликовал это - если вам нужны индексы элементов во время цикла через массив, вы можете сделать это:
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
не нужно цитировать, потому что они просто цифры; некоторые предложили бы цитировать их в любом случае, но это просто личное предпочтение.)
попробуйте это. Он работает и тестируется.
for k in "${array[@]}"
do
echo $k
done
# For accessing with the echo command: echo ${array[0]}, ${array[1]}
простой способ :
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}
Это самое простое решение, предполагая, что в именах нет пробелов:
listofNames = "A B C D"
for databaseName in $listOfNames; do
echo $databaseName
done
Я перебираю массив моих проектов для git pull
обновление:
#!/bin/sh
projects="
web
ios
android
"
for project in $projects do
cd $HOME/develop/$project && git pull
end