Команда test проверяет выполнение некоторого условия. С использованием этой (встроенной) команды формируются операторы выбора и цикла языка shell.
Два возможных формата команды:
test условие или
[ условие ]
мы будем пользоваться вторым вариантом, т.е. вместо того, чтобы писать перед условием слово "test", будем заключать условие в скобки, что более привычно для программистов.
На самом деле shell будет распознавать эту команду по открывающей скобке "[",. как слову(!), соответствующему команде "test". Между скобками и содержащимся в них условием обязательно должны быть пробелы. •
Пробелы должны быть и между значениями и символом сравнения или операции (как, кстати, и в команде "expr"). Не путать с противоположным требованием для присваивания значений переменным.
В shell используются условия различных "типов".
УСЛОВИЯ ПРОВЕРКИ ФАЙЛОВ:
-f file файл "file" является обычным файлом;
-d file файл "file" - каталог;
-с file файл "file" - специальный файл;
-r file имеется разрешение на чтение файла "file";
-w file имеется разрешение на запись в файл "file"
-s file файл "file" не пустой. Примеры. Вводя с клавиатуры командные строки в первом случае получим подтверждение (код завершения "О"), а во втором - опровержение (код завершения "1"). "specific" - имя существующего файла.
[ -f specific ] ; echo $?
[ -d specific ] ; echo $?
УСЛОВИЯ ПРОВЕРКИ СТРОК:
str1 = str2 строки "strl" и "str2" совпадают;
str1 != str2 строки "str1" и "str2" не совпадают;
-n str1 строка "str1" существует (непустая);
-z str1 строка "str1" не существует (пустая). Примеры.
x="who is who"; export х; [ "who is who" = "$x" ]; echo $?
x=abc ; export х ; [ abc = "$x" ] ; echo $?
x=abc ; export х ; [ -n "$x" ] ; echo $?
x="" ; export х ; [ -n "$x" ] ; echo $?
Команда "test" дает значение "истина" (т.е. код завершения "О") и просто если в скобках стоит непустое слово.
[ privet ] ; echo $?
[ ] ; echo $?
Кроме того, существуют два стандартных значения условия, которые могут использоваться вместо условия (для этого не нужны скобки).
true ; echo $?
false ; echo $?
УСЛОВИЯ СРАВНЕНИЯ ЦЕЛЫХ ЧИСЕЛ: х -eq у "х" равно "у", х -nе у "х" неравно "у", х -gt у "х" больше "у", х -ge у "х" больше или равно "у", х -lt у "х" меньше "у", х -1е у "х" меньше или равно "у". То есть в данном случае команда "test" воспринимает строки символов как целые (!) числа. Поэтому во всех остальных случаях "нулевому" значению соответствует пустая строка. В данном же случае, если надо обнулить переменную, скажем, "х", то это достигается присваиванием "х=0". Примеры.
x=3.21 ; export х ; [ 3.21 -eq "$x" ] ; echo $? "[": integer expression expected before -eq
x=321 ; export х ; [ 123 -lt "$x" ] ; echo $? 0
СЛОЖНЫЕ УСЛОВИЯ:
Реализуются с помощью типовых логических операций:
! (not) инвертирует значение кода завершения.
-о (or) соответствует логическому "ИЛИ".
-a (and) соответствует логическому "И". ПРЕДУПРЕЖДЕНИЕ. Не забывайте о пробелах. Примеры.
[ ! privet ] ; echo $?
x=privet; export х; [ "$x" -a -f specific ] ; echo $? 0
x="";export х; [ "$x" -a -f specific ] ; echo $? 1
x=""; export х; [ "$x" -a -f specific -o privet ] ; echo $? 0
x="";export х; [ "$x" -a -f specific -o ! privet ] ; echo $? 1
1.2. Условный оператор "if
B общем случае оператор "if имеет структуру
if условие
then список
[elif условие then список]
[else список] fi
Здесь "elif' сокращенный вариант от "else if может быть использован наряду с полным, т.е. допускается вложение произвольного числа операторов "if (как и других операторов). Разумеется "список" в каждом случае должен быть осмысленный и допустимый в данном контексте.
Конструкции
[elif условие
then список] и
[else список]
не являются обязательными (в данном случае для указания на необязательность конструкций использованы квадратные скобки - не путать с квадратными скобками команды "test"!). Самая усеченная структура этого оператора if условие
then список fi
если выполнено условие (как правило это ком получен код завершения "О", то выполняется "список", иначе он пропускается. Обратите внимание, что структура обязательно завершается служебным словом "fi". Число "fi", естественно, всегда должно соответствовать числу "if. Примеры.
Пусть написан расчет "if-1" if [ $1 -gt $2 ] then pwd
else echo $0 : Hello! Fi Тогда вызов расчета
if-1 12 11 даст
/home/sae/STUDY/SHELL a
if-1 12 13 даст
if-1 : Hello!
Возможно использовать в условии то свойство shell, что команды могут выдавать различный код завершения. Это напоминает приемы программирования на Си. Пусть расчет "if-2" будет
if а=`ехрг "$1" : "$2"` then echo then a=$a code=$? else echo else a=$a code=$? Fi тогда вызов
if-2 by by даст
then a=2 code=0 a
if-2 by be даст
else a=0 code=l Еще пример на вложенность
###
# if-З: Оценка достижений
echo -n " А какую оценку получил на экзамене?: "
read z
if [ $z - 5 ]
then echo Молодец !
elif [ $z=4 ]
then echo Все равно молодец ! elif [ $z = 3 ] then echo Все равно ! elif [ $z = 2 ]
then echo Bсe ! else echo ! f i
Можно обратить внимание на то, что желательно использовать сдвиги при записи программ, чтобы лучше выделить вложенность структур.
1.3.Оператор вызова ("case")
Оператор выбора "case" имеет структуру:
case строка in шаблон) список команд;; шаблон) список команд;;
esac
Здесь "case" "in" и "esac" - служебные слова. "Строка" (это может быть и один символ) сравнивается с "шаблоном". Затем выполняется "список команд" выбранной строки. Непривычным будет служебное слово "esac", но оно необходимо для завершения структуры.
Пример.
###
# case-1: Структура "case".
# Уже рассматривавшийся в связи со
# структурой "if" пример проще и
# нагляднее можно реализовать с
# помощью структуры "case", echo -n " А какую оценку получил на экзамене?: "
read z
case $z in
5) echo Молодец ! ;;
4) echo Все равно молодец ! ;;
3) echo Все равно ! ; ;
2) echo Все ! ;;
*) echo ! esac
Непривычно выглядят в конце строк выбора ";;", но написать здесь ";" было бы ошибкой. Для каждой альтернативы может быть выполнено несколько команд. Если эти команды будут записаны в одну строку, то символ ";" будет использоваться как разделитель команд.
Обычно последняя строка выбора имеет шаблон "*", что в структуре "case" означает "любое значение". Эта строка выбирается, если не произошло совпадение значения переменной (здесь $z) ни с одним из ранее записанных шаблонов, ограниченных скобкой ")". Значения просматриваются в порядке записи.
При вызове "case-2 Hello" на экран будет выведено:
Холдинг: Привет!
А при вызове "case-2 HELLO" на экран будет выведено:
Нет такой фирмы
Коль скоро слово "case" переводится как "выбор", тоэто как бы намек на то, что можно эту структуру использовать для реализации простейших меню.
###
# case-З: Реализация меню с помощью команды "case"
echo "Назовите файл, а затем (через пробел) наберите цифру, соответствующую требуемой обработке:
1 - отсортировать
2 - выдать на экран
3 - определить число строк " read х у # х - имя файла, у - что сделать case $y in
1) sort < $x ; ;
2) cat < $х
3) wc -1 < $х
*) echo "
Мы не знаем
такой команды ! " ; ; esac
Разумеется, желания могут быть более сложные и на месте отдельных команд могут быть последовательности команд или вызовы более сложных расчетов.
Напишем команду "case-4", которая добавляет информацию к файлу, указанного первым параметром (если параметр один), со стандартного входа, либо (если 2 параметра) из файла, указанного в качестве первого параметра:
###
# case-4: Добавление в файл.
# Использование стандартной переменной.
# "$#" - число параметров при вводе расчета
# "»" - перенаправление с добавлением в
файл
case $# in
1) cat » $1
2) cat » $2 < $1
*) echo "Формат: case-4 [откуда] куда" ;;
esac
"$1" (при "$#=1") - это имя файла, в который происходит добавление со стандартного входа.
"$1" и "$2" (при $#=2) - это имена файлов , из которого ("$1") и в который ("$2") добавлять.
Во всех других случаях (*) выдается сообщение о том, каким должен быть правильный формат команды.
1.4.Оператор цикла с перечислением ("for")
Оператор цикла "for" имеет структуру: for имя [in список значений] do
список команд
done
где "for" - служебное слово определяющее тип цикла, "do" и "done" - служебные слова, выделяющие тело цикла. Не забывайте про "done"! Фрагмент "in список значений" может отсутствовать. Пусть команда "lsort" представлена командным файлом
for i in f1 f2 f3
do
proc-sort $i
done
B этом примере имя "i" играет роль параметра цикла. Это имя можно рассматривать как shell-переменную, которой последовательно присваиваются перечисленные значения (i=f1, i=f2, i=f3), и выполняется в цикле команда "procsort".
Часто используется форма "for i in *", означающая "для всех файлов текущего каталога".
Пусть "proc-sort" в свою очередь представляется командным
файлом
cat $1 | sort I tee /dev/lp > ${l}_sorted
т.е. последовательно сортируются указанные файлы, результаты сортировки выводятся на печать ("/dev/lp") и направляются в файлы f1_sorted f2_sorted и f3_sorted
Можно сделать более универсальной команду "lsort", если не фиксировать перечень файлов в команде, а передавать произвольное их число параметрами.
Тогда головная программа будет следующей:
for i
do
proc-sort $i
done
Здесь отсутствие после "i" служебного слова "in" с перечислением имен говорит о том , что список поступает через параметры команды. Результат предыдущего примера можно получить, набрав
lsort f1 f2 f3
Усложним ранее рассматривавшуюся задачу (под именем "case-2") определения холдинга фирмы. Теперь можно при вызове указывать произвольное количество фирм. При отсутствии в структуре оператора "for" фрагмента "in список значений", значения берутся из параметров вызывающей команды.
###
# holding: Справочник.
# Для различных фирм по имени выдается
# название холдинга, в который она входит
for i
do
case $i in
ONE|TWO|THREE) echo Холдинг: ZERO ; ;
МММ|WWW) echo Холдинг: Not-Net ;;
Hi|Hello|Howdoing) echo Холдинг: Привет! ;;
*) echo Нет такой фирмы ;;
esac
done
При вызове "holding Hello HELLO ONE" на экране будет: