Литмир - Электронная Библиотека
Содержание  
A
A

Программа

lint
, с одной стороны, указывает на недостатки в вашей программе, а с другой выдает много не относящихся к делу сообщений, которые мы выше опустили, и нужен некоторый опыт, чтобы уметь разбираться, какие из них необходимы, а какие следует игнорировать. Однако имеет смысл постараться, так как
lint
помогает обнаружить некоторые ошибки, которые человек увидеть практически не может. После длительного редактирования всегда стоит запустить
lint
и убедиться в том, что каждое выдаваемое этой программой предупреждение вам понятно.

6.7 Пример:

zap

Программа

zap
, которая избирательно уничтожает процессы, отличается от той, что была представлена в виде файла
shell
в гл. 5. Главная проблема данной версии скорость. Она создает много процессов и поэтому работает медленно, что недопустимо для программы, уничтожающей процессы с ошибками. Если переписать
zap
на Си, ее быстродействие повысится. Мы, однако, снова воспользуемся
ps
, чтобы найти информацию о процессе. Это намного легче, чем выуживать информацию из ядра, и, кроме того, мы имеем переносимый вариант. Программа
zap
открывает программный канал, входной поток для которого берется из
ps
, и читает из него, как из файла. Функция
popen(3)
аналогична
fopen
, за исключением того, что первый аргумент является командой, а не именем файла. То же самое справедливо и для
pclose
, но здесь она нам не нужна.

/* zap: interactive process killer */

#include <stdio.h>

#include <signal.h>

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
, определенный в
<signal.h>
, не может быть перехвачен или проигнорирован. Вы можете вспомнить пятую главу, где его численное значение равно девяти, но лучше использовать символические константы из файлов макроопределений, чем включать в свои программы загадочные числа.

Если аргументы отсутствуют,

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>i</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
и ждать ее окончания
83
{"b":"248117","o":1}