《UNIX环境高级编程(第3版)》——1.9 信号

    xiaoxiao2023-06-27  168

    本节书摘来自异步社区《UNIX环境高级编程(第3版)》一书中的第1章,第1.9节,作者:【美】W. Richard Stevens , Stephen A.Rago著,更多章节内容可以访问云栖社区“异步社区”公众号查看

    1.9 信号

    信号(signal)用于通知进程发生了某种情况。例如,若某一进程执行除法操作,其除数为0,则将名为SIGFPE(浮点异常)的信号发送给该进程。进程有以下3种处理信号的方式。

    (1)忽略信号。有些信号表示硬件异常,例如,除以0或访问进程地址空间以外的存储单元等,因为这些异常产生的后果不确定,所以不推荐使用这种处理方式。

    (2)按系统默认方式处理。对于除数为0,系统默认方式是终止该进程。

    (3)提供一个函数,信号发生时调用该函数,这被称为捕捉该信号。通过提供自编的函数,我们就能知道什么时候产生了信号,并按期望的方式处理它。

    很多情况都会产生信号。终端键盘上有两种产生信号的方法,分别称为中断键(interrupt key,通常是Delete键或Ctrl+C)和退出键(quit key,通常是Ctrl+),它们被用于中断当前运行的进程。另一种产生信号的方法是调用kill函数。在一个进程中调用此函数就可向另一个进程发送一个信号。当然这样做也有些限制:当向一个进程发送信号时,我们必须是那个进程的所有者或者是超级用户。

    实例回忆一下基本的shell实例(见图1-7程序)。如果调用此程序,然后按下中断键,则执行此程序的进程终止。产生这种后果的原因是:对于此信号(SIGINT)的系统默认动作是终止进程。该进程没有告诉系统内核应该如何处理此信号,所以系统按默认方式终止该进程。

    为了能捕捉到此信号,程序需要调用signal函数,其中指定了当产生SIGINT信号时要调用的函数的名字。函数名为sig_int,当其被调用时,只是打印一条消息,然后打印一个新提示符。在图1-7程序中添加了11行,构成了图1-10程序(添加的11行以行首的+号指示)。

    #include "apue.h"  #include <sys/wait.h> + static void sig_int(int);  /* our signal-catching function */ +  int  main(void)  {     char  buf[MAXLINE]; /* from apue.h */     pid_t pid;     int  status; +   if (signal(SIGINT, sig_int) == SIG_ERR) +      err_sys("signal error"); +      printf("%% "); /* print prompt (printf requires %% to print %) */     while (fgets(buf, MAXLINE, stdin) != NULL) {       if (buf[strlen(buf) - 1] == '\n')          buf[strlen(buf) - 1] = 0; /* replace newline with null */       if ((pid = fork()) < 0) {          err_sys("fork error");       } else if (pid == 0) { /* child */          execlp(buf, buf, (char *)0);          err_ret("couldn't execute: %s", buf);          exit(127);       }       /* parent */       if ((pid = waitpid(pid, &status, 0)) < 0)          err_sys("waitpid error");       printf("%% ");     }     exit(0);  } + + void + sig_int(int signo) + { +   printf("interrupt\n%% "); + }

    图1-10 从标准输入读命令并执行

    因为大多数重要的应用程序都对信号进行处理,所以第10章将详细介绍信号。

    相关资源:用TCP/IP进行网际互联 第三卷:客户-服务器编程与应用(Linux/POSIX套接字版)--详细书签版
    最新回复(0)