Более сложные выражения могут быть образованы с помощью логических операторов:
!<i>выражение</i>
| Истинно, если выражение ложно (оператор NOT) |
<i>выражение1</i> -а <i>выражение2</i>
| Истинно, если оба выражения истинны (оператор AND) |
<i>выражение1</i> -o <i>выражение2</i>
| Истинно, если хотя бы одно из выражений истинно (оператор OR) |
Приведем несколько примеров использования выражений.
Фрагмент скрипта, используемый при регистрации нового пользователя. Скрипт проверяет наличие в домашнем каталоге инициализационного скрипта .profile и в случае его отсутствия копирует шаблон:
if [ ! -f $НОМЕ/.profile ]
then
echo "файла .profile не существует - скопируем шаблон"
cp /usr/lib/mkuser/sh/profile $НОМЕ/.profile
fi
Фрагмент скрипта, проверяющего наличие новой почты в почтовом ящике пользователя
if [ -s $MAIL ]
then
echo "Пришла почта"
fi
Фрагмент скрипта инициализации системы — запуска "суперсервера" Internet inetd(1M). Если исполняемый файл /etc/inetd существует, он запускается на выполнение.
if [ -х /etc/inetd ]
then
/etc/inetd
echo "запущен сервер inetd"
fi
Фрагмент скрипта, анализирующий ввод пользователя, сохраненный в переменной ANSW. Если пользователь ввел 'N' или 'n', скрипт завершает свою работу.
if [ "$ANSW" = "N" -о "$ANSW" = "n" ]
then
exit
fi
Циклы
Язык программирования Bourne shell имеет несколько операторов цикла. Приведем их синтаксис:
1)
while <i>условие</i>
do
<i> command1</i>
<i> command2</i>
...
done
2)
until <i>условие</i>
do
<i> command1</i>
<i> command2</i>
...
done
3)
for <i>var</i> in <i>список</i>
do
<i> command1</i>
<i> command2</i>
...
done
С помощью оператора while команды
<i>command1</i>
,
<i>command2</i>
и т.д. будут выполняться, пока
<i>условие</i>
не станет ложным. Как и в случае с оператором
if,
<i>условие</i>
генерируется кодом возврата команды, например,
<i>test</i>
.
В случае оператора until команды
<i>command1</i>
,
<i>command2</i>
и т.д. будут выполняться, пока
<i>условие</i>
не станет истинным.
Оператор for обеспечивает выполнение цикла столько раз, сколько слов в
<i>списке</i>
. При этом переменная
<i>var</i>
последовательно принимает значения, равные словам из списка. Список может формироваться различными способами, например как вывод некоторой команды (`
имя_команды_формирующей_список`) или с помощью шаблонов shell.
В другой форме for, когда список отсутствует, переменная var принимает значения позиционных параметров, переданных скрипту.
Чтобы наглядно представить себе приведенные операторы, обратимся к конкретным примерам.
Например, скрипт монтирования всех файловых систем /etc/mounall для системы Solaris 2.5 включает в себя их проверку, исходя из данных, указанных в файле /etc/vfsck. При этом используется оператор while.
#
cat /etc/vfsck |
while read special fsckdev mountp fstype fsckpass automnt mntopts
# Построчно считывает записи файла vfsck и присваивает переменным spe-
# cial, fsckdev и т.д. значения соответствующих конфигурационных полей.
do
case $special in
'# ' * | '' ) # Игнорируем комментарии
continue ;;
'-') # Игнорируем строки, не требующие действия
continue ;;
esac
# Последовательно проверяем файловые системы с помощью утилиты
# /usv/sbin/fsck
/usr/sbin/fsck -m -F $fstype $fsckdev >/dev/null 2>&1
...
done
Скрипт очистки давно не используемых файлов во временных каталогах (обычно он запускается при загрузке системы) использует оператор for.
for dir in /tmp /usr/tmp /home/tmp
do
find $dir ! -type d -atime +7 -exec rm {} \;
done
При этом удаляются все файлы в указанных каталогах (/tmp, /usr/tmp и /home/tmp), последний доступ к которым осуществлялся более недели назад.
Селекторы
Оператор case предоставляет удобную форму селектора:
case <i>слово</i> in
<i>шаблон1</i>)