Linux: Полное руководство — страница 32 из 98

>. Если введено меньше значений, чем нужно, оставшимся переменным будет присвоена пустая строка. Если больше, то весь остаток строки будет присвоен последней из перечисленных переменных.

8.2. Подстановки

Кроме подстановки обычных переменных (п.3.4.4) и раскрытия шаблонов имен файлов (п.3.4.5), оболочка bash умеет выполнять еще и такие подстановки:

Подстановка тильды:

~ заменяется на имя домашнего каталога пользователя, запустившего сценарий;

~+ заменяется на путь к текущему каталогу;

~- заменяется на путь к предыдущему каталогу.

Раскрытие скобок:

строка1{строка2,строка3...}строкаN
заменяется на:
строка1строка2строкаN строка1строка3строкаN ...

Пробелов внутри скобок быть не должно. Эта функция полезна, когда нужно применить одну команду к нескольким файлам, не подходящим под общий шаблон:

$ cat /home/den/linuxbook/{intro,param,subst} > glava8

Подстановка арифметических выражений:

$((выражение))
или
$[выражение]
— эквивалентные формы записи. Внутри выражения выполняются подстановки параметров. Приоритет арифметических операций — обычный, подробнее см.
man bash,
секция ARITHMETIC EVALUATION.

Например, количество часов, прошедшее с момента запуски оболочки, можно подсчитать командой:

$ echo $(( $SECONDS/3600 ))

8.3. Массивы

Интерпретатор bash поддерживает одномерные массивы с неограниченным числом элементов. В других оболочках существуют определенные ограничения на массивы, например, в ksh максимальное число элементов массива ограничено 1024 элементами.

Нумерация элементов начинается с нуля. Тип элементов массива, как и тип параметров, строковый. Присвоить значение элементу массива можно с помощью такой конструкции:

Имя_массива[индекс] = значение
, например:

$ weekday[0]=Понедельник

$ weekday[4]=Пятница

Обратиться к значению элемента массива можно следующим образом:

${имя_массива[индекс]}

Например, вывести значение первого элемента массива можно так:

$ echo ${weekday[0]}

Обратиться ко всем элементам массива сразу можно с помощью одного из выражений:

${имя_массива[*]}
или
${имя_массива[@]}
, например:

$echo ${weekday[*]}

Понедельник Пятница

Второе выражение нужно использовать, если значение хотя бы одного элемента массива может содержать пробелы.

Можно инициализировать массив целиком: команда

$ weekday=(Пн Вт Ср Чт Пт Сб Вс)

эквивалентна списку

$ weekday[0]=Пн; weekday[1] =Вт; ... weekday[6] =Вс

Эти способы инициализации массивов могут применяться как в bash, так и в ksh и в других командных оболочках. А следующий способ работает только в bash:

$ holidays= ([0]=Sunday [6]=Saturday)

8.4. Управляющие структуры

Напоминаю (п.3.4.7), что список команд — это одиночная команда, конвейер или последовательность команд/конвейеров, разделенных одним из операторов: ; && ||, завершенная точкой с запятой. Не забывайте ставить точку с запятой даже после одиночной команды, терять ее — типичная ошибка начинающих программистов.

В синтаксис следующих команд квадратные скобки не входят: они ограничивают необязательные выражения.

8.4.1. Условные операторы

Оператор варианта case

Синтаксис:

case значение in

 [шаблон1) список1;;

 шаблон2 | шаблон3) список2;; ]

esac

Ищется первый шаблон, совпадающий со значением. Если он найден, то выполняется соответствующий ему список команд, завершенный двумя символами «;». Шаблон и список разделяются символом «)». Одному списку команд может соответствовать несколько шаблонов, тогда они разделяются символом «|».

В шаблонах могут использоваться метасимволы *, ? и [] (о подстановке метасимволов сказано в п.3.4.5). С их помощью можно организовать инструкцию, действующую как default в операторе switch языка С.

Порядок сравнения шаблонов со значением не определен. Первое совпадение прекращает дальнейшее сравнение и приводит к выполнению соответствующего списка команд и выходу из структуры case — поведение, больше похожее на поведение оператора case в языке Паскаль, чем на switch в языке С.

Листинг 8.1. Пример использования оператора case

echo "Ошибка. Кому переслать протокол?"

echo "Начальнику: b"

echo "Коллегам: с"

echo "Сам разберусь: any key"

read answer

case $answer in

b|B) mail -s "error log" boss < error.log;;

c|C) mail -s 'Help!!! error log' -c ivan den < error.log;;

*) echo "OK. Exiting";

 exit;;

esac

Обратите внимание на апострофы вместо кавычек во втором списке: они экранируют подстроку «!!» от подстановки предыдущей команды.

Условный оператор if

Синтаксис:

if список1 then

 список2

[elif список3 then

 список4]

[else

 список5]

fi

Эта конструкция работает так же, как в других языках программирования. Сначала выполняются команды из

списка1
. Если этот список выполнен успешно, то есть с кодом завершения 0, то выполняется
список2
, если нет — то список, стоящий после очередного elif. При невозможности выполнить список команд, стоящий после очередного then, выполняется список, стоящий после else.

Можно использовать сокращенный вариант, только if-then-fi:

$ if [ $? -ne 0 ]; then echo "Ошибка. Смотри протокол"; fi;

Оператор test и условные выражения

В вышеприведенной команде вместо анализа кода завершения списка использована проверка условия. Две формы такой проверки эквивалентны: встроенная команда test и [условие]. Например, для проверки существования файла можно написать

test -e <файл>

или

[ -е <файл> ]

Если вместо слова test используются квадратные скобки, они обязательно должны быть отделены от аргументов пробелом, потому что на самом деле «[» — это название команды, а «]» — обязательный последний аргумент этой команды.

В случае истинности условия команда test возвращает код успешного завершения, то есть 0; в случае ложности — код ошибки 1 (не спутайте с обычными языками программирования, где 1 — другое имя для true!).

Команда test может проверять и строку на пустоту: непустая строка считается выполнением условия и приводит к коду завершения 0. Пример:

$ test $USER; echo $?

0

$ test $VAR_not_set_yet; echo $?

1

Условные выражения можно комбинировать с помощью обычных логических операций:

!(выражение)
— отрицание;

выражение1 -а выражение2
— логическое И (and);

выражение1 -о выражение2
— логическое ИЛИ (or).

Элементарные условные выражения перечислены в таблицах 8.2 и 8.3. Полный список их можно получить по команде

help test
.


Основные условные выражения для файлов Таблица 8.2

ВыражениеИстинно, если
-d файлфайл существует и является каталогом
-е файлфайл существует
-f файлфайл существует и является обычным файлом
-L файлфайл существует и является символической ссылкой
-r файлфайл существует и доступен для чтения
-w файлфайл существует и доступен для записи
-x файлфайл существует и является исполняемым
-s файлфайл существует и его размер больше 0
-N файлфайл существует и изменился со времени последнего чтения
файл1 -nt файл2время модификации файла1 позже (newer than), чем файла2
файл1 -ot файл2время модификации файла1 раньше (older than), чем файла2
файл1 -ef файл2файл1 — это жесткая ссылка на файл2

Элементарные условные выражение для сравнения строк Таблица 8.3

ВыражениеИстинно, если
-z строкадлина строки равна 0
-n строкадлина строки не равна 0
стр1 == стр2строки совпадают
стр1 !== стр2строки не совпадают
стр1 < стр2строка1 предшествует строке2 в лексикографическом порядке. Алфавит соответствует текущей локали
стр1 > стр2строка1 следует за строкой2 в лексикографическом порядке

Арифметическое условное выражение имеет формат

arg1 OP arg2
, где
arg1
и
arg2
 — целые числа, a
OP
— одна из операций:

-eq — равно;

-ne — не равно;

-lt — меньше;

-le — меньше или равно;

-gt — больше;

-ge — больше или равно.

Таким образом, можно переписать предыдущий пример с использованием оператора if:

Листинг 8.2. Пример использования оператора if

echo "Ошибка. Кому переслать протокол?"

echo "Начальнику: b"

echo "Коллегам: c"

echo "Сам разберусь; any key"

read answer

if [ "$answer" == "b" -o "$answer" == "B" ]; then

 mail -s "error log" boss < error.log;

elif [ "$answer" == "с" -о "$answer" == "C" ]; then

 mail -s 'Help!!! error log' -c ivan den < error.log;

else

 echo "OK. Exiting"; exit;

fi

8.4.2. Операторы цикла

Командные интерпретаторы bash и ksh поддерживают циклы типа for, while, until и select, а интерпретатор sh — только циклы for и while.

Оператор цикла с перечислением for

Синтаксис:

for переменная [in шаблон]

do

 список

done

В результате подстановки шаблона получается список слов. Переменная получает значение первого слова из этого списка, и выполняется список команд, стоящий между do и done. Затем переменная получает значение очередного слова из списка слов, и снова выполняется список команд. Повторение прекращается по исчерпании слов в списке. Отсутствие конструкции [in шаблон] эквивалентно записи in $@.

Список слов можно сформировать и вручную:

$ for day in Mon Tue Wed Thu Fri; do echo "План работы на $day:"; cat $day.plan; done

Еще раз напомню, что любой список в bash нужно заканчивать точкой с запятой.

Пример использования цикла с перечислением: допустим, у вас не хочет собираться некий программный пакет — он рассчитывал, что имена заголовочных файлов в некотором каталоге имеют расширение .h, а у вас они такого расширения не имеют (установлена другая версия библиотеки). При этом содержание этих файлов его устраивает. Так создадим символические ссылки, чтобы он нашел заголовки по знакомому имени:

$ cd /путь/к/каталогу/include

$ for name in *; do ln -s $name $name.h; done

Оператор цикла с условием while

Синтаксис:

while список1

do

 список2

done

Оператор выполняет

список1
и в случае его успешного завершения (нулевого кода возврата) —
список2
. Процедура повторяется до тех пор, пока результат выполнения списка1 не станет ненулевым. Например:

$ i = 10

$ while [ $i -gt 0 ]; do

> echo $i...

> i=${($i-1))

> done; echo 'BANG!!!'

$

Примечание

Тот же самый обратный отсчет можно реализовать и с помощью цикла for, если у вас установлена утилита seq, печатающая последовательность (sequence) чисел с заданным шагом:

$ for i in `seq 10 -1 0`; do echo $i...; done; echo 'BANG!!!'

Оператор цикла с инверсным условием until

Синтаксис:

until список1

do

 список2

done

Оператор выполняет

список1
и, если он выполнен неуспешно (код возврата ненулевой), то выполняет
список2
. Процедура повторяется до тех пор, пока результат выполнения
списка1
не станет нулевым.

Оператор цикла с выбором select

Синтаксис:

select переменная [in шаблон]

do

 список

done

В результате подстановки шаблона получается список слов. К этим словам оператор добавляет порядковые номера и выводит весь набор в стандартный поток ошибок (stderr). Если шаблон опущен, то вместо него используется список позиционных параметров. После этого оболочка выводит приглашение и считывает строку из стандартного потока ввода (stdin). Если строка содержит номер, соответствующий какому-либо слову из списка, то переменная получает это слово в качестве значения. Если в строке подходящего номера нет, то значением переменной становится пустая строка. После этого выполняется список команд, и процедура повторяется до тех пор, пока в строке ввода не встретится символ конца файла (введите Ctrl+D) или пока в списке команд не встретится команда break или return.

Этот оператор полезен для создания нумерованных пунктов меню. Например, у меня в каталоге

~/temp
есть три файла:
proto.txt
,
file.txt
и
README
. В листинге 8.3. приведен фрагмент сценария, позволяющего быстро просмотреть любой из них.

Листинг 8.3. Пример использования оператора select

echo "Выберите файл для просмотра:"

select file in ~/temp/* Quit;

do

 if [ -f $file ]; then cat $file;

 else break;

 fi

done

Запустив этот сценарий, я увижу на экране:

Выберите файл для просмотра:

1) /home/den/temp/file.txt

2) /home/den/temp/proto.txt

3) /home/den/temp/README

4) Quit

#?

Последняя строка — это приглашение, устанавливаемое переменной окружения PS3.

8.5. Условная подстановка параметров