Брайан Керниган - UNIX — универсальная среда программирования
- Название:UNIX — универсальная среда программирования
- Автор:
- Жанр:
- Издательство:Финансы и статистика
- Год:1992
- Город:Москва
- ISBN:5-289-00253-4
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Брайан Керниган - UNIX — универсальная среда программирования краткое содержание
В книге американских авторов — разработчиков операционной системы UNIX — блестяще решена проблема автоматизации деятельности программиста, системной поддержки его творчества, выходящей за рамки языков программирования. Профессионалам открыт богатый "встроенный" арсенал системы UNIX. Многочисленными примерами иллюстрировано использование языка управления заданиями shell.
Для программистов-пользователей операционной системы UNIX.
UNIX — универсальная среда программирования - читать онлайн бесплатно ознакомительный отрывок
Интервал:
Закладка:
$ lint pick.с
...
fprintf, arg. 1 несовместим "llib-lc"(69) :: "pick.c"(28)
...
$
Это означает, что первый аргумент в стандартной библиотеке определен иначе, чем в строке 28 вашей программы. Таким образом дана точная информация о том, что неверно.
Программа lint
, с одной стороны, указывает на недостатки в вашей программе, а с другой выдает много не относящихся к делу сообщений, которые мы выше опустили, и нужен некоторый опыт, чтобы уметь разбираться, какие из них необходимы, а какие следует игнорировать. Однако имеет смысл постараться, так как lint
помогает обнаружить некоторые ошибки, которые человек увидеть практически не может. После длительного редактирования всегда стоит запустить lint
и убедиться в том, что каждое выдаваемое этой программой предупреждение вам понятно.
6.7 Пример: zap
Программа zap
, которая избирательно уничтожает процессы, отличается от той, что была представлена в виде файла shell
в гл. 5. Главная проблема данной версии скорость. Она создает много процессов и поэтому работает медленно, что недопустимо для программы, уничтожающей процессы с ошибками. Если переписать zap
на Си, ее быстродействие повысится. Мы, однако, снова воспользуемся ps
, чтобы найти информацию о процессе. Это намного легче, чем выуживать информацию из ядра, и, кроме того, мы имеем переносимый вариант. Программа zap
открывает программный канал, входной поток для которого берется из ps
, и читает из него, как из файла. Функция popen(3)
аналогична fopen
, за исключением того, что первый аргумент является командой, а не именем файла. То же самое справедливо и для pclose
, но здесь она нам не нужна.
/* zap: interactive process killer */
#include
#include
char *progname; /* program name for error message */
char *ps = "ps -ag"; /* system dependent */
main(argc, argv)
int argc;
char *argv[];
{
FILE *fin, *popen();
char buf[BUFSIZ];
int pid;
progname = argv[0];
if ((fin = popen(ps, "r")) == NULL) {
fprintf(stderr, "%s: can't run %s\n", progname, ps);
exit(1);
}
fgets(buf, sizeof buf, fin); /* get header line */
fprintf(stderr, "%s", buf);
while (fgets(buf, sizeof buf, fin) != NULL)
if (argc == 1 || strindex(buf, argv[1]) >= 0) {
buf[strlen(buf)-1] = '\0'; /* suppress \n */
fprintf(stderr, "%s? ", buf);
if (ttyin() == 'y') {
sscanf(buf, "%d", &pid);
kill(pid, SIGKILL);
}
}
exit(0);
}
Мы писали программу, чтобы использовать ps -ag
(этот флаг системно зависим), но если вы не являетесь привилегированным пользователем, то можете уничтожать лишь свои собственные процессы.
Первый вызов fgets
выбирает заголовок из ps
; интересно выяснить, что случится, если попытаться уничтожить "процесс", соответствующий данному заголовку.
Функция sscanf
представляет собой член семейства scanf(3)
для форматного преобразования входной строки. Она преобразует строку, а не файл. Вызов kill
из системы посылает специальный сигнал процессу; сигнал SIGKILL
, определенный в , не может быть перехвачен или проигнорирован. Вы можете вспомнить пятую главу, где его численное значение равно девяти, но лучше использовать символические константы из файлов макроопределений, чем включать в свои программы загадочные числа.
Если аргументы отсутствуют, zap
предоставляет каждую строку выходного потока ps
как возможность для выбора. При наличии аргумента zap
предлагает только те выходные строки ps
, которые ему соответствуют. Функция strindex(s1, s2)
проверяет, соответствует ли аргумент какой-либо части строки выходного потока ps
, используя strncmp
(см. табл. 6.2). Функция strindex
возвращает позицию s2
в s1
или -1, если ее там нет.
strindex(s, t) /* return index of t in s, -1 if none */
char *s, *t;
{
int i, n;
n = strlen(t);
for (i = 0; s[i] != '\0'; i++)
if (strncmp(s+i, t, n) == 0)
return i;
return -1;
}
В табл. 6.4 представлены широко используемые функции из стандартной библиотеки ввода вывода.
fp=fopen(s, mode) |
Открыть файл s ; значения mode "r" , "w" , "a" соответствуют чтению, записи и добавлению (при ошибке возвращается NULL) |
c=gets(fp) |
Читать символ: getchar() это getc(stdin) |
putc(c, fp) |
Записать символ: putchar(c) это putc(c, stdout) |
ungetc(c, fp) |
Вернуть символ во входной файл fp ; можно вернуть не более одного символа за раз |
scanf(fmt, a1, ...) |
Читать символы из stdin в a1 , ... в соответствии с fmt . Каждый a i должен быть указателем |
fscanf(fp,...) |
Читать из файла fp |
sscanf(s,...) |
Читать из строки s |
printf(fmt, a1, ...) |
Форматировать a1 , ... в соответствии с fmt ; печатать в stdout |
fprintf(fp, ...) |
Печатать ... в файл fp |
sprintf(s, ...) |
Печатать ... в строку s |
fqets(s, n, fp) |
Читать не более n символов в s из fp (возвращается NULL по концу файла) |
fputs(s, fp) |
Печатать строку s в файл fp |
fflush(fp) |
Занести буферизованные данные выходного потока в файл fp |
fclose(fp) |
Закрыть файл fp |
fp=popen(s, mode) |
Открыть программный канал для команды s (см. fopen ) |
pclose(fp) |
Закрыть программный канал fp |
system(s) |
Запустить команду s и ждать ее окончания |
Таблица 6.4: Полезные стандартные функции ввода-вывода
Модифицируйте zap
так, чтобы можно было применять любое число аргументов. В настоящем виде zap
высвечивает на экране строку, соответствующую выбранному варианту. Будет она делать это? Если нет, модифицируйте программу соответствующим образом. Подсказка: getpid(2)
.
Постройте fgrep(1)
на основе strindex
. Сравните время работы при сложных поисках, например 10 слов на документ. Почему fgrep
выполняется быстрее?
6.8 Диалоговая программа сравнения файлов: idiff
Поддерживать две чем-то отличающиеся версии файла, каждая из которых содержит часть нужного вам файла, довольно распространенная проблема. Зачастую она возникает в тех случаях, когда изменения вносятся независимо двумя разными людьми. Программа diff
подскажет вам, чем различаются файлы, но вы не получите никакой помощи, если захотите выбрать какую-то информацию из одного файла, а какую-то из другого.
В этом разделе мы напишем программу idiff
(диалоговая diff
), которая предоставляет пользователю каждую порцию выходного потока diff
и предлагает ему возможность выбора фрагментов "от и до" или их редактирования. Программа idiff
помещает выбранные фрагменты в соответствующем порядке в файл idiff.out
. Допустим, даны такие два файла:
Интервал:
Закладка: