Linux программирование в примерах
10 {11 char buf[200], *cp;12 int offset;1314 /* Пройти через это испытание , чтобы избежать fprintf(). */15 strcpy(buf, "handler: caught signal ");16 cp = buf + strlen(buf); /* cp указывает на завершающий '\0' */17 if (signum > 100) /* маловероятно */18 offset = 3;19 else if (signum > 10)20 offset = 2;21 else22 offset = 1;23 cp += offset;2425 *cp-- = '\0'; /* завершить строку */26 while (signum >0) { /* work backwards, filling in digits */27 *cp-- = (signum % 10) + '0';28 signum /= 10;29 }30 strcat(buf, "\n");31 (void)write(2, buf, strlen(buf));32 }3334 /* main --- установить обработку сигнала и войти в бесконечный цикл */3536 int main(void)37 {38 (void)signal(SIGINT, handler);3940 for(;;)41 pause(); /* ждать сигнал, см. далее в главе */4243 return 0;44 }Строки 9–22 определяют функцию обработки сигнала (остроумно названную
[106]). Все, что эта функция делает, — выводит номер перехваченного сигнала и возвращается. Для вывода этого сообщения она выполняет множество ручной работы, посколькуhandler()не является «безопасной» для вызова из обработчика сигнала. (Вскоре это будет описано в разделе 10.4.6 «Дополнительные предостережения».)fprintf()Функция
устанавливает обработчик сигнала (строка 38), а затем входит в бесконечный цикл (строки 40–41). Вот что происходит при запуске:main()$ <b>ssh solaris.example.com</b>/* Зарегистрироваться на доступной системе Solaris */Last login: Fri Sep 19 04:33:25 2003 from 4.3.2.1.Sun Microsystems Inc. SunOS 5.9 Generic May 2002$ <b>gcc ch10-catchint.c</b> /* Откомпилировать программу */$ <b>a.out</b> /* Запустить ее */<b>^C </b>handler: caught signal 2 /* Набрать ^C, вызывается обработчик */<b>^C</b> /* Попробовать снова, но на этот раз... */$ /* Программа завершается */Поскольку V7 и другие традиционные системы восстанавливают действие сигнала по умолчанию, поэтому когда вы хотите снова получить сигнал в будущем, функция обработчика должна немедленно переустановить саму себя:
void handler(int signum) {char buf[200], *cp;int offset;(void)signal(signum, handler); /* переустановить обработчик *//* ...оставшаяся часть функции как прежде... */}10.4.2. BSD и GNU/Linux
BSD 4.2 изменила способ работы
. [107] На системах BSD обработчик сигнала после его возвращения остается на месте. Системы GNU/Linux следуют поведению BSD. Вот что происходит под GNU/Linux:signal()$ <b>ch10-catchint</b> /* Запустить программу */handler: caught signal 2 /* Набираем ^C, вызывается обработчик */handler: caught signal 2 /* И снова... */handler: caught signal 2 /* И снова! */handler: caught signal 2 /* Помогите! */handler: caught signal 2 /* Как нам это остановить?! */Quit (core dumped) /* ^\, генерирует SIGQUIT. Bay */На системе BSD или GNU/Linux обработчик сигнала не должен дополнительно использовать '
' для переустановки обработчика. Однако, лишний вызов не причиняет никакого вреда, поэтому сохраняется статус-кво.signal(signum, handler)В действительности, POSIX предоставляет функцию
, которая идентичнаbsd_signal()за тем исключением, что она гарантирует, что обработчик сигнала останется установленным:signal()#include <signal.h> /* XSI, устаревает */void (*bsd_signal(int sig, void (*func)(int)))(int);Это устраняет проблемы переносимости. Если вы знаете, что ваша программа будет работать лишь на системах POSIX, вы можете воспользоваться
вместоbsd_signal().signal()Одно предостережение — эта функция также помечена как «устаревающая», что означает возможность отказа от нее в будущем стандарте. На практике, даже если от нее откажутся, поставщики скорее всего долгое время будут ее поддерживать. (Как мы увидим, функция API POSIX
предоставляет достаточно возможностей для написания рабочей версии, если это вам нужно.)sigaction()10.4.3. Игнорирование сигналов
Более практично, когда вызывается обработчик сигнала, это означает, что программа должна завершиться и выйти. Было бы раздражающим, если бы большинство программ по получении
выводили бы сообщение и продолжали работу; смысл сигнала в том, что они должны остановиться!SIGINTНапример, рассмотрите программу
.sort, возможно, создала любое число временных файлов для использования на промежуточных этапах процесса сортировки. По полученииsort,SIGINTдолжна удалить временные файлы и выйти. Вот упрощенная версия обработчика сигнала из GNU Coreutilssort:sort.c