Вход/Регистрация
Linux программирование в примерах
вернуться

Роббинс Арнольд

Шрифт:

6 #include <sys/time.h>

7

8 /* handler --- обрабатывает SIGALRM */

9

10 void handler(int signo)

11 {

12 static const char msg[] = "\n*** Timer expired, you lose ***\n";

13

14 assert(signo == SIGALRM);

15

16 write(2, msg, sizeof(msg) - 1);

17 exit(1);

18 }

19

20 /* main --- установить таймер, прочесть данные с тайм-аутом */

21

22 int main(void)

23 {

24 struct itimerval tval;

25 char string[BUFSIZ];

26

27 timerclear(&tval.it_interval); /* нулевой интервал означает не сбрасывать таймер */

28 timerclear(&tval.it_value);

29

30 tval.it_value.tv_sec = 10; /* тайм-аут 10 секунд */

31

32 (void)signal(SIGALRM, handler);

33

34

35 printf("You have ten seconds to enter\nyour name, rank, and serial number: ");

36 (void)setitimer(ITIMER_REAL, &tval, NULL);

37 if (fgets(string, sizeof string, stdin) != NULL) {

38 (void)setitimer(ITIMER_REAL, NULL, NULL); /* выключить таймер */

39 /* обработать оставшиеся данные, вывод диагностики для иллюстрации */

40 printf("I'm glad you are being cooperative.\n");

41 } else

42 printf("\nEOF, eh? We won't give up so easily'\n");

43

44 exit(0);

45 }

Строки 10–18 представляют обработчик сигнала для

SIGALRM
; вызов
assert
гарантирует, что обработчик сигнала был установлен соответствующим образом. Тело обработчика выводит сообщение и выходит, но оно может делать что-нибудь более подходящее для крупномасштабной программы.

В функции

main
строки 27–28 очищают два члена
struct timeval
структуры
struct itimerval.tval
. Затем строка 30 устанавливает тайм-аут в 10 секунд. Установка
tval.it_interval
в 0 означает, что нет повторяющегося сигнала; он срабатывает лишь однажды. Строка 32 устанавливает обработчик сигнала, а строка 34 выводит приглашение.

Строка 36 устанавливает таймер, а строки 37–42 выводят соответствующие сообщения, основываясь на действиях пользователя. Реальная программа выполняла бы в этот момент свою задачу. Важно здесь обратить внимание на строку 38, которая отменяет таймер, поскольку были введены действительные данные.

ЗАМЕЧАНИЕ. Между строками 37 и 38 имеется намеренное состояние гонки. Все дело в том, что если пользователь не вводит строку в течение отведенного таймером времени, будет доставлен сигнал, и обработчик сигнала выведет сообщение «you lose».

Вот три успешных запуска программы:

$ ch14-timers /* Первый запуск, ничего не вводится */

You have ten seconds to enter

your name, rank, and serial number:

*** Timer expired, you lose ***

$ ch14-timers /* Второй запуск, ввод данных */

You have ten seconds to enter

your name, rank, and serial number: Jamas Kirk, Starfleet Captain, 1234

I'm glad you are being cooperative.

$ ch14-timers /* Третий запуск, ввод EOF (^D) */

You have ten seconds to enter

your name, rank, and serial number: ^D

EOF, eh? We won't give up so easily!

POSIX оставляет неопределенным, как интервальные таймеры взаимодействуют с функцией

sleep
, если вообще взаимодействуют. GLIBC не использует для реализации
sleep
функцию
alarm
, поэтому на системах GNU/Linux
sleep
не взаимодействует с интервальным таймером. Однако, для переносимых программ, вы не можете делать такое предположение.

  • Читать дальше
  • 1
  • ...
  • 247
  • 248
  • 249
  • 250
  • 251
  • 252
  • 253
  • 254
  • 255
  • 256
  • 257
  • ...

Ебукер (ebooker) – онлайн-библиотека на русском языке. Книги доступны онлайн, без утомительной регистрации. Огромный выбор и удобный дизайн, позволяющий читать без проблем. Добавляйте сайт в закладки! Все произведения загружаются пользователями: если считаете, что ваши авторские права нарушены – используйте форму обратной связи.

Полезные ссылки

  • Моя полка

Контакты

  • chitat.ebooker@gmail.com

Подпишитесь на рассылку: