Shell script — различия между версиями
Andy (обсуждение | вклад) (→case) |
Andy (обсуждение | вклад) (→case) |
||
(не показана 1 промежуточная версия этого же участника) | |||
Строка 508: | Строка 508: | ||
действие | действие | ||
... | ... | ||
+ | ;; | ||
условие2) | условие2) | ||
действие | действие | ||
действие | действие | ||
... | ... | ||
+ | ;; | ||
*) | *) | ||
действие | действие | ||
действие | действие | ||
... | ... | ||
+ | ;; | ||
esac | esac | ||
</syntaxhighlight> | </syntaxhighlight> | ||
Условие <code>*)</code> означает "во всех остальных случаях". Пример: | Условие <code>*)</code> означает "во всех остальных случаях". Пример: | ||
+ | <syntaxhighlight lang="bash"> | ||
+ | [root@vm-01 ~]# cat test.sh | ||
+ | #!/bin/bash | ||
+ | |||
+ | case "$1" in | ||
+ | |||
+ | "0") | ||
+ | echo "It is a zero!: $1" | ||
+ | ;; | ||
+ | |||
+ | "100") | ||
+ | echo "Wow! You have entered: $1" | ||
+ | ;; | ||
+ | |||
+ | *) | ||
+ | if [ -z "$1" ]; then | ||
+ | echo "You have entered nothing" | ||
+ | exit 1 | ||
+ | else | ||
+ | echo "You have entered: $1 " | ||
+ | fi | ||
+ | ;; | ||
+ | |||
+ | |||
+ | esac | ||
+ | |||
+ | exit 0 | ||
+ | [root@vm-01 ~]# ./test.sh | ||
+ | You have entered nothing | ||
+ | [root@vm-01 ~]# ./test.sh 2 | ||
+ | You have entered: 2 | ||
+ | [root@vm-01 ~]# ./test.sh 0 | ||
+ | It is a zero!: 0 | ||
+ | [root@vm-01 ~]# ./test.sh 100 | ||
+ | Wow! You have entered: 100 | ||
+ | [root@vm-01 ~]# | ||
+ | </syntaxhighlight> | ||
== Ссылки == | == Ссылки == |
Текущая версия на 12:37, 4 апреля 2018
Содержание
Основы создания скриптов на bash
Предварительные требования
- Виртуальная машина с двумя сетевыми интерфейсами
- Установленные пакеты:
bash
,bash-completion
,vim-minimal
,vim-enhanced
Общая информация
Любой скрипт, начинается с "магической строки" - сигнатуры, которая говорит командному интерпретатору как обрабатывать файл.
Для скриптов, такая последовательность носит называние she-bang
(сокращение от sharp-bang
):
#!
Соответственно, если файл должен обрабатывать bash
, то это необходимо указать после "магической строки". То есть, she-bang
будет иметь следующий вид:
#!/bin/bash
Если скрипт предназначен для пользователя, то должны быть выставлен необходимый владелец файла, и права на запуск скрипта 755
. Поскольку
скрипт является выполняемым, то по первым литерам названия, коммандный интерпертатор производит поиск названия в переменной $PATH
,
соответственно, лучше выбрать имя для скрипта, которое не совпадает с имеющейся утилитой. Для просмотра переменной $PATH
надо выполнить следующую команду:
[root@vm-01 ~]# echo $PATH
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin
[root@vm-01 ~]#
Поскольку текущая директория /root
отсутствует в переменной $PATH
, то запускать скрипты необходимо следующим образом:
[root@vm-01 ~]# ls -lahi test.sh
33969143 -rwxr-xr-x. 1 root root 34 Mar 26 10:38 test.sh
[root@vm-01 ~]# ./test.sh
Hello, world!
[root@vm-01 ~]#
Вывод на экран
Скрипт test.sh
использует комманду echo
для отображения текстовой строки. По-умолчанию отображение производится
на стандартный поток вывода (STDOUT
):
[root@vm-01 ~]# cat test.sh
#!/bin/bash
echo "Hello, world!"
[root@vm-01 ~]#
Хорошим тоном считается выводить ошибки на поток ошибок (STDERR
):
[root@vm-01 ~]# cat test.sh
#!/bin/bash
echo "Hello, world!"
echo "Something has happend! Can not proceed!" >&2
[root@vm-01 ~]#
Запустим скрипт таким образом, что бы поток ошибок направлялся в файл:
[root@vm-01 ~]# ./test.sh 2>test.log
Hello, world!
[root@vm-01 ~]# cat test.log
Something has happend! Can not proceed!
[root@vm-01 ~]#
При выводе на экран, следует учесть, что некоторые символы могут иметь специальное значение, в следствие чего, использование таких
символов может привести к результатам отличным от ожидаемого. Например, если необходимо узнать содержимое PATH
, то следует
написать так:
[root@vm-01 ~]# cat test.sh
#!/bin/bash
echo "$PATH"
[root@vm-01 ~]# ./test.sh
/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin
[root@vm-01 ~]#
Если необходимо вывести именно строку $PATH, то следует использовать символы экранирования (двойные кавычки "
, одинарные кавычки '
, либо обратный слэш \
):
[root@vm-01 ~]# cat test.sh
#!/bin/bash
echo "\$PATH"
echo '$PATH'
[root@vm-01 ~]# ./test.sh
$PATH
$PATH
[root@vm-01 ~]#
Следует заметить, что двойные кавчки экранируют не все спецсимволы, поэтому если необходимо вывести какую-либо информацию как текст, то следует использовать одинарные кавычки, которые интерпретируют
любую включенную в них информацию как текст! Обратный слэш (\
) экранирует только символ следующий за ним!
Использование переменных
Переменная - это область памяти, которая хранит некое значение. К этой области памяти можно обращаться по имени. Название переменной не может начинаться с цифры! Например:
[root@vm-01 ~]# cat test.sh
#!/bin/bash
one=1
echo "This is number $one"
[root@vm-01 ~]# ./test.sh
This is number 1
[root@vm-01 ~]#
Хорошим тоном считается заключать значения, присваиваемые переменными в двойные кавычки, к примеру, потому, что в значении могут содержаться пробелы, а пробелы интерпретируются
bash
'ем как разделитель слова. Так же, хорошим тоном считается заключение переменных в фигурные скобки, так как переменные могут конкатенироваться непосредственно
друг за другом, поэтому, во избежании получения неожиданных результатов, переменные следует записывать как ${one}
.
Пример конкатенцации переменных:
[root@vm-01 ~]# cat test.sh
#!/bin/bash
one="1"
two="2"
echo "This is number twelve: ${one}${two}"
[root@vm-01 ~]# ./test.sh
This is number twelve: 12
[root@vm-01 ~]#
Расширения командного интерпретатора
Кроме переменных, bash
позволяет использовать подстановку команд и арифметические операции
Подстановка команд
Подстановка команд осуществляется путем помещения команды в обратные кавычки `
, либо путем указания переменной в круглых скобках ()
. Например:
[root@vm-01 ~]# cat test.sh
#!/bin/bash
cli="`which bash`"
curl="$(which curl)"
echo "My bash is here: ${cli} and curl is here: ${curl}"
[root@vm-01 ~]# ./test.sh
My bash is here: /usr/bin/bash and curl is here: /usr/bin/curl
[root@vm-01 ~]#
Второй способ предпочтительнее, так как он поддерживает вложенность команд. Например:
[root@vm-01 ~]# cat test.sh
#!/bin/bash
count="$($(which ls) -l *.conf | $(which wc) -l)"
echo "Amount of conf files in /root directory: ${count}"
[root@vm-01 ~]# ./test.sh
Amount of conf files in /root directory: 4
[root@vm-01 ~]#
Арифметика
Арифметические операции осуществляются путем помещения команды в квадратные скобки ([]
), которым предшествует знак $
. Например:
[root@vm-01 ~]# cat test.sh
#!/bin/bash
count1=$[ 3 * 3 ]
count2=$[2+2]
echo "${count1}"
echo "${count2}"
[root@vm-01 ~]# ./test.sh
9
4
[root@vm-01 ~]#
Часто применяемые арифметические действия и их значения
Оператор | Значение |
---|---|
<переменная>++ | постинкремент переменной |
<переменная>-- | постдекремент переменной |
++<переменная> | преинкремент переменной |
--<переменная> | предекремент переменной |
- | вычитание, унарный минус |
+ | сложение, унарный плюс |
** | возведение в степень |
* | умножение |
/ | деление |
% | остаток от деления |
Если в арифметическом выражении есть несколько действий, то bash
выполняет такие операции в соответствии с правилами математики.
Таблица приоритета арифметических действий:
Оператор | Значение |
---|---|
<переменная>++, <переменная>-- | постинкремент и постинкремент переменных |
++<переменная>, --<переменная> | преинкремент и предекремент переменных |
-, + | унарный минус и плюс |
** | возведение в степень |
*, /, % | умножение, деление, остаток от деления |
+, - | сложение, вычитание |
Циклы bash
Для повторяемых действий в bash
имеются циклы for
, while
и until
Использование цикла for
Синтаксис цикла выглядит следующим образом:
for переменная in список; do команды; done
Например:
[root@vm-01 ~]# cat test.sh
#!/bin/bash
count=0
for i in $(ls /root/*.conf); do
count=$[count+1]
echo "${count}"
done
[root@vm-01 ~]# ./test.sh
1
2
3
4
[root@vm-01 ~]
Специальные переменные bash
Переменная | Значение |
---|---|
$? | код завершения программы (как правило: 0 - успешное завершение, все отилчное от 0 - ошибка) |
$0 | имя самой программы |
$1..$9 | аргументы скрипта |
$@ | список аргументов ($1,$2,$3..) |
$* | список аргументов, рассматриваемых как единый аргумент ("$1 $2 $3..") |
$# | количество переданных аргументов |
[root@vm-01 ~]# cat test.sh
#!/bin/bash
count="$#"
for i in "$@"; do
echo "Current argument is: $i"
echo "Total amount of args: ${count}"
echo "Whole arg string: $*"
done
[root@vm-01 ~]# ./test.sh 1 2 3 4 5 6 7 8 9
Current argument is: 1
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 2
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 3
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 4
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 5
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 6
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 7
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 8
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
Current argument is: 9
Total amount of args: 9
Whole arg string: 1 2 3 4 5 6 7 8 9
[root@vm-01 ~]# echo $?
0
[root@vm-01 ~]#
Сравнения
Для сравнения переменных в bash
, используется либо команда test
, либо внутренние команды [[]]
.
Числовые сравнения
Оператор | Значение | Пример |
---|---|---|
-eq | равно | [ "$a" -eq "$b" ] |
-ne | неравно | [ "$a" -ne "$b" ] |
-gt | больше чем | [ "$a" -gt "$b" ] |
-ge | больше или равно | [ "$a" -ge "$b" ] |
-lt | меньше чем | [ "$a" -lt "$b" ] |
-le | меньше или равно | [ "$a" -le "$b" ] |
Строковые сравнения
Оператор | Значение | Пример |
---|---|---|
= | равно | [ "$a" = "$b" ] |
== | равно | [ "$a" == "$b" ] |
!= | неравно | [ "$a" != "$b" ] |
Логические операторы AND и OR
Оператор | Значение |
---|---|
&& | Следующая команда выполняется только, если результат предыдущий был успешен (истина) |
|| | Следующая команда выполняется, если результат предыдущий был неуспешен (ложь) |
Условия
if/then
Общие правила построения условий if/then
условий:
if условие; then
действие
действие
...
fi
Например:
[root@vm-01 ~]# cat test.sh
#!/bin/bash
count="$#"
if [ "$count" -gt 1 ]; then
echo "Number of args greater than one"
fi
for i in "$@"; do
echo "Current argument is: $i"
echo "Total amount of args: ${count}"
echo "Whole arg string: $*"
done
exit 0
[root@vm-01 ~]# ./test.sh
[root@vm-01 ~]# ./test.sh 1
Current argument is: 1
Total amount of args: 1
Whole arg string: 1
[root@vm-01 ~]# ./test.sh 1 2
Number of args greater than one
Current argument is: 1
Total amount of args: 2
Whole arg string: 1 2
Current argument is: 2
Total amount of args: 2
Whole arg string: 1 2
[root@vm-01 ~]#
if/then/else
Общие правила построения условий if/then/else
условий:
if условие; then
действие
действие
...
else
действие
действие
...
fi
Например:
[root@vm-01 ~]# cat test.sh
#!/bin/bash
count="$#"
if [ "$count" == 0 ]; then
echo "You did not enter any args!"
exit 1
else
for i in "$@"; do
echo "Current argument is: $i"
echo "Total amount of args: ${count}"
echo "Whole arg string: $*"
done
fi
exit 0
[root@vm-01 ~]# ./test.sh
You did not enter any args!
[root@vm-01 ~]# echo $?
1
[root@vm-01 ~]# ./test.sh 0 1
Current argument is: 0
Total amount of args: 2
Whole arg string: 0 1
Current argument is: 1
Total amount of args: 2
Whole arg string: 0 1
[root@vm-01 ~]# echo $?
0
[root@vm-01 ~]#
if/then/elif/then/else
Расширение условия выглядит следующим образом:
if условие; then
действие
действие
...
elif условие; then
действие
действие
...
else
действие
действие
...
fi
Например
[root@vm-01 ~]# cat test.sh
#!/bin/bash
count="$#"
if [ "$count" == 0 ]; then
echo "You did not enter any args!"
exit 1
elif [ "$count" -gt 5 ]; then
echo "You have entered too many args!"
exit 2
else
for i in "$@"; do
echo "Current argument is: $i"
echo "Total amount of args: ${count}"
echo "Whole arg string: $*"
done
fi
exit 0
[root@vm-01 ~]# ./test.sh
You did not enter any args!
[root@vm-01 ~]# ./test.sh 1 2 3 4 5 6
You have entered too many args!
[root@vm-01 ~]# ./test.sh 1 2 3 4 5
Current argument is: 1
Total amount of args: 5
Whole arg string: 1 2 3 4 5
Current argument is: 2
Total amount of args: 5
Whole arg string: 1 2 3 4 5
Current argument is: 3
Total amount of args: 5
Whole arg string: 1 2 3 4 5
Current argument is: 4
Total amount of args: 5
Whole arg string: 1 2 3 4 5
Current argument is: 5
Total amount of args: 5
Whole arg string: 1 2 3 4 5
[root@vm-01 ~]#
Хоть и конструкций elif
может быть сколько угодно, не рекомендуется прибегать к их большому количеству,
так как это ухудшает читабельность кода.
case
Вместо elif
, в тех случаях, когда требуется обширная логика условий, следует
применять case
case значение in
условие1)
действие
действие
...
;;
условие2)
действие
действие
...
;;
*)
действие
действие
...
;;
esac
Условие *)
означает "во всех остальных случаях". Пример:
[root@vm-01 ~]# cat test.sh
#!/bin/bash
case "$1" in
"0")
echo "It is a zero!: $1"
;;
"100")
echo "Wow! You have entered: $1"
;;
*)
if [ -z "$1" ]; then
echo "You have entered nothing"
exit 1
else
echo "You have entered: $1 "
fi
;;
esac
exit 0
[root@vm-01 ~]# ./test.sh
You have entered nothing
[root@vm-01 ~]# ./test.sh 2
You have entered: 2
[root@vm-01 ~]# ./test.sh 0
It is a zero!: 0
[root@vm-01 ~]# ./test.sh 100
Wow! You have entered: 100
[root@vm-01 ~]#
Ссылки
Интерпретатор командного языка shell
Программирование на языке командного интерпретатора - shell
Bash
Advanced Bash-Scripting Guide