В этой программе нужно контролировать процесс разбиения интерпретатором строк на аргументы, чтобы только символ перевода строки разделял соседние "слова".
Внутренняя переменная интерпретатора
IFS
(internal field separator — внутренний разделитель полей) представляет собой строку символов, которая разделяет слова в списке аргументов, находящихся в знаках слабого ударения или циклах
for
. Обычно
IFS
содержит пробелы, символы табуляции и конца строки, но мы можем заменить ее на что-либо нужное, например просто на символ перевода строки:
Мы здесь кое-что добавили: необязательный аргумент, обозначающий сигнал (обратите внимание на то, что
SIG
будет неопределенным, а значит, должен рассматриваться как пустая строка, если аргумент не задан), а также
egrep
вместо
grep
, чтобы разрешить более сложные шаблоны типа
'sleep | date'
. Первая команда
echo
выдает столбец из заголовков выходных данных команды
ps
.
Вас может заинтересовать, почему эта команда называется
zap
, а не просто
kill
. Основная причина заключается в том, что в отличие от случая с командой
cal
мы не даем действительно новой команды
kill
:
zap
по необходимости является диалоговой командой, с одной стороны, а с другой — мы хотим сохранить имя
kill
для настоящей команды. К тому же
zap
чрезвычайно медленна из-за накладных расходов на все дополнительные программы, хотя самую длинную по времени реализации команду
ps
все равно нужно выполнять. В следующей главе будет продемонстрировано более эффективное решение.
Упражнение 5.23
Измените команду
zap
так, чтобы она, выдавая заголовки из команды
ps
, была
не чувствительна к изменениям в формате вывода
ps
. Насколько это усложнит программу?
5.7 Команда
pick
: пробелы или аргументы
Вы уже достаточно подготовлены для того, чтобы написать команду
pick
на языке
shell
. Единственным новым средством является механизм чтения входного потока пользователя. Встроенная команда интерпретатора
read
читает одну строку текста из стандартного входного потока и присваивает ее (без перевода строки) в качестве значения указанной переменной:
$ read greeting
hello, world
Вводим новое значение для приветствия
$ echo $greeting
hello, world
$
Самым типичным примером использования команды
read
в файле
.profile
служит установка значений переменных среды при входе в систему, прежде всего установка переменных интерпретатора типа
TERM
.
Команда
read
может читать только из стандартного входного потока; его нельзя даже переключить. Ни одну из встроенных команд интерпретатора (в отличие от основных структур управления типа
for
) нельзя переключить с помощью операций
>
или
<
:
$ read greeting </etc/passwd
goodbye
Тем не менее надо ввести значение
illegal io
Сейчас shell сообщает об ошибке
$ echo $greeting
greeting получает введенное значение,
goodbye
а не значение из файла
$
Это можно считать ошибкой интерпретатора, но такова жизнь. К счастью, можно предусмотреть переключение в цикле, охватывающем команду
read
, что является основным принципом реализации команды
pick
:
# pick: select arguments
PATH=/bin:/usr/bin
for i # for each argument
do
echo -n "$i? " >/dev/tty
read response
case $response in
y*) echo $i ;;
q*) break
esac
done </dev/tty
Обращение
echo -n
подавляет заключительный символ перевода строки, так что переменную
response
можно вывести на той же строке, что и приглашение. Конечно, приглашения выдаются на устройство
/dev/tty
, поскольку стандартный выходной поток, по всей вероятности, не выводится на терминал.
Оператор
break
заимствован из языка Си: он завершает выполнение самого внутреннего цикла, в нашем случае
for
, когда вводится
q
. Мы выбрали символ
q
как сигнал прекращения процесса выбора потому, что это легко сделать, потенциально удобно и не противоречит другим программам.
Интересно поэкспериментировать с пробелами в аргументах для команды