$ cat news
# news: print news files, version 1
HOME=. # debugging only
cd . # place holder for /usr/news
for i in `ls -t * $HOME/.news_time`
do
case $i in
*/.news_time) break ;;
*) echo news: $i
esac
done
touch $HOME/.news_time
$ touch .news-time
$ touch x
$ touch y
$ news
news: y
news: x
$
Команда
touch
заменяет время последней модификации файла, заданного в качестве аргумента, на настоящее время, не подвергая сам файл модификации. Для отладки мы даем только эхо имен файлов новостей, а не печатаем их. Цикл завершается при обнаружении
news_time
, тем самым перечисляются только файлы со свежими новостями. Заметьте, что символ
*
в операторе
case
может быть сопоставлен с
/
, что недопустимо для шаблонов имен файлов. А что будет, если
news_time
не существует?
$ rm .news_time
$ news
$
Отсутствие ответа удивляет и является ошибочным. Это вызвано тем, что когда команда
ls
не находит файл, она выдает соответствующее сообщение в стандартный выходной поток прежде, чем вывести какую-либо информацию о существующих файлах. Такая ситуация, безусловно, ошибочна — диагностические сообщения должны передаваться в стандартный файл диагностики. Но мы можем обнаружить эту ситуацию в цикле и переключить стандартный файл диагностики на стандартный выходной поток, так что все версии будут работать одинаково. (Данная проблема решена в новой версии, но мы рассмотрели ее, чтобы проиллюстрировать, как легко устранить недоделки.)
$ cat news
# news: print news files, version 2
HOME=. # debugging only
cd . # place holder for /usr/news
IFS='
' # just a newline
for i in `ls -t * $HOME/.news_time 2>&1`
do
case $i in
*' not found') ;;
*/.news_time) break ;;
*) echo news: $i ;;
esac
done
touch $HOME/.news_time
$ news
news: news
news: y
news: x
$
Мы должны были установить
IFS
равным символу конца строки, чтобы сообщение
./.news_time not found
не распознавалось как три слова.
Команда
news
должна выводить на печать файлы новостей, а не создавать эхо их имен. Полезно знать, кто и когда послал сообщение, поэтому мы воспользуемся командами
set
и
ls -l
для вывода заголовка перед самим сообщением:
$ ls -l news
-rwxrwxrwx 1 you 208 Oct 1 12:05 news
$ set `ls -l news`
-rwxrwxrwx: bad option(s)
Что-то неправильно!
$
Это один из тех случаев, когда взаимозаменяемость программы и данных на языке
shell
имеет значение. Команда
set
"ругается", потому что ее аргумент ("
-rwxrwxrwx
") начинается с минуса и, следовательно, выглядит как флаг. Очевидным (хотя и неэлегантным) решением было бы предварить аргумент обычным символом:
$ set X`ls -l news`
$ echo "news: ($3) $5 $6 $7"
news: (you) Oct 1 12:05
$
Здесь представлен разумный формат с указанием автора и даты сообщения вместе с именем файла. Приведем окончательный вариант команды
news
:
# news: print news files, final version
PATH=/bin:/usr/bin
IFS='
' # just a newline
cd /usr/news
for i in `ls -t * $HOME/.news_time 2>&1`
do
IFS=' '
case $i in
*' not found') ;;
*/.news_time) break ;;
*) set X`ls -l $i`
echo "
$i: ($3) $5 $6 $7
"
cat $i
esac
done
touch $HOME/.news_time
Дополнительные символы перевода строк разделяют в заголовке при печати фрагменты новостей. Первым значением
IFS
является символ перевода строки, поэтому сообщение
not found
из вывода первой команды
ls
(если оно есть) рассматривается как один аргумент. Во втором случае переменной
IFS
присваивается пробел, поэтому вывод второй команды
ls
разбивается на несколько аргументов.
Упражнение 5.27
Добавьте в команду news флаг
-n
("notify" — извещение), чтобы сообщать о новостях, но не печатать их, и не выполняйте
touch .news_time
. Эту команду можно поместить в ваш файл
.profile
.