23 24 25 26 27 28 29
30 31
$
Досадно, что месяц нужно задавать числом, и к тому же, как оказалось, команда
cal 10
выдает календарь на весь 10-й год, а не на октябрь текущего года. Поэтому всегда следует указывать год, если вы хотите получить календарь на один месяц.Указанные выше неудобства связаны с тем, что взаимодействие пользователя с программой было реализовано без привлечения команды
cal
. Вы можете изменить характер этого взаимодействия, не меняя самой программы. Если поместить команду в ваш собственный каталог bin
, то возможен более удобный способ перевода аргументов в те, которые нужны настоящей команде cal
. Вы можете даже вызывать свою версию команды, и тогда вам меньше придется запоминать.Первый шаг разработки — определить функции усовершенствованной команды
cal
. В основном мы хотим от нее разумного поведения. Месяц нужно распознавать по названию. При наличии двух аргументов она должна делать то же, что делала прежняя версия, за исключением перевода названия месяца в его номер. В случае одного аргумента следует печатать календарь месяца или года (в зависимости от того, что вам требуется), а при отсутствии аргументов — календарь текущего месяца, так как большей частью именно для этого и обращаются к команде. Поэтому задача сводится к тому, чтобы определить, сколько аргументов задано, и преобразовать их в те параметры, которые требуются стандартной команде cal
.Язык
shell
имеет оператор case
, который успешно применяется в таких ситуациях:case слово in
шаблон) команды ;;
шаблон) команды ;;
...
esac
В операторе
case
слово сравнивается поочередно со всеми шаблонами от начала до конца и выполняются команды, связанные с первым (и только первым) шаблоном, соответствующим слову. Шаблоны составляются по правилам соответствия шаблонов, которые в некоторой степени обобщают правила задания имен файлов. Каждое действие завершается двумя символами ;;
(для последнего варианта можно обойтись без ;;
, но обычно мы ставим их для удобства редактирования).В нашей версии команды определяется число заданных аргументов, обрабатываются названия месяцев, затем происходит обращение к настоящей команде
cal
. В переменной интерпретатора $#
хранится число аргументов, с которыми была вызвана программа; другие специальные переменные интерпретатора перечислены в табл. 5.1.$#
Число аргументов $*
Все аргументы, передаваемые интерпретатору $@
Аналогично $*
; см. разд. 5.7 $-
Флаги, передаваемые интерпретатору $?
Возвращение значения последней выполненной команды $$
Номер процесса интерпретатора $!
Номер процесса последней команды, запущенной с помощью &
$НOМЕ
Аргумент, принятый по умолчанию для команды cd
$IFS
Список символов, разделяющих слова в аргументах $MAIL
Файл, изменение которого приводит к появлению сообщения "you have a mail" ("У вас есть почта") $PATH
Список каталогов, в которых осуществляется поиск команд $PS1
Строка приглашение, по умолчанию принята '$'
$PS2
Строка приглашение при продолжении командной строки, по умолчанию принята '>'
Таблица 5.1: Встроенные переменные интерпретатора
$ cat cal
# cal: nicer interface to /usr/bin/cal
case $# in
0) set `date`; m=$2; y=$6 ;; # no args: use today
1) m=$l; set `date`; y=$6 ;; #1 arg: use this year
*) m=$1; y=$2 ;; #2 args: month and year
esac
case $m in
jan*|Jan*) m=1 ;;
feb*|Feb*) m=2 ;;
mar*|Mar*) m=3 ;;
apr*|Apr*) m=4 ;;
may*|May*) m=5 ;;
jun*|Jun*) m=6 ;;
jul*|Jul*) m=7 ;;
aug*|Aug*) m=8 ;;
sep*|Sep*) m=9 ;;
oct*|Oct*) m=10 ;;
nov*|Nov*) m=11 ;;
dec*|Dec*) m=12 ;;
[1-9]|10|11|12) ;; # numeric month
*) y=$m; m="" ;; # plain year
esac
/usr/bin/cal $m $y # run the real one
$
В первом операторе case проверяется число аргументов
$#
и выбирается подходящее действие. Последний шаблон в этом операторе задает вариант, выбираемый по умолчанию; если число аргументов не 0 и не 1, будет выполнен последний вариант. (Поскольку шаблоны просматриваются по порядку, вариант по умолчанию должен быть последним.) При наличии двух аргументов m
и y
принимают значение месяца и года, и наша команда cal
должна выполняться как исходная команда.Первый оператор
case
включает пару нетривиальных строк, содержащихset `date`
Хотя это сразу и не очевидно, легко установить действие команды, запустив ее:
$ date
Sat Oct 1 06:05:18 EDT 1983
$ set `date`
$ echo $1
Sat
$ echo $4
06:05:20
$
Итак, мы имеем дело с встроенной командой интерпретатора, возможности которой многообразны. При отсутствии аргументов она выдает, как указывалось в гл. 3, значения переменных окружения. В случае обычных аргументов переопределяются значения
$1
, $2
и т.д. Поэтому set `date`
присваивает $1
— день недели, $2
— название месяца и т.д. Таким образом, при отсутствии аргументов в первом case
месяц и год устанавливаются из текущей даты. Если был задан один аргумент, он используется в качестве месяца, а год берется из текущей даты.Команда
set
имеет также несколько флагов, из которых наиболее часто используются флаги -v
и -х
— для отключения эха команд при обработке их интерпретатором. Такое отключение может оказаться необходимым в процессе отладки сложных программ на языке shell
.Теперь осталось только перевести значение месяца, если оно представлено в строковом виде, в число. Это делается с помощью второго оператора
case
, который практически очевиден. Единственный нюанс состоит в том, что символ |
в шаблонах оператора case
, как и в команде egrep
, означает альтернативу: малый|большой
соответствует варианту "малый" или "большой". Конечно, эти варианты можно было бы задать с помощью [jJ]an*
и т.д. Программа допускает задание месяца строчными буквами, поскольку большинство команд работает с входным потоком, где данные записаны строчными буквами (иногда первая буква — прописная), поскольку так выглядит вывод команды date
. Правила сопоставления шаблонов приведены в табл. 5.2.*
Задает любую строку, включая пустую ?
Задает любой одиночный символ [ccc]
Задает любой из символов в ccc [a-d0-3]
эквивалентно [abcd0123]
"..."
Задает в точности ...
; кавычки защищают от специальных символов. Аналогично действует '...'
\c