UNP学习笔记之三-POSIX Signal Handling

说道多进程的svr网络模型,免不了会用到一些进程间通信的工具,在对简单的基于fork的并发服务器进行介绍以后,这里简单地介绍了一下信号处理:

  1. 1、信号也称为软件中断,信号的产生总是异步的,进程无法提前知道信号会什么时候发生。
  2. 2、信号可以在进程间发送,也可以是内核发送给进程。

3、每一个信号产生时,都会触发一个相关的行为(函数)执行,这个行为(函数)我们可以使用sigaction函数定义.

4、对于信号所关联的行为,我们有以下3种选择:

当一个特定的信号发生时,可以执行一个自定义的函数,这个函数称之为“信号处理器”,这个函数执行的操作我们称之为捕获了一个信号。但是信号SIGKILL和SIGSTOP不能被捕获。大多数的程序可以使用sigaction来指定函数。少部分信号需要一些附加的操作才能捕获,例如:SIGIO,SIGPOLL,SIGURG。

 

设置行为值为SIG_IGN时,在信号发生时将不做任何操作,但是信号SIGKILL和SIGSTOP不能被忽略。

可以将行为默认设置为SIG_DFL。在信号发生时,默认的信号行为大多数是终止整个进程;还有一些进程会在进程的执行目录下面生成core文件;有少量信号的默认行为是被忽略掉的(SIGCHLD和SIGURG)。

有信号处理的svr

#include        "unp.h"

int main(int argc, char **argv)
{
        int                                     listenfd, connfd;
        pid_t                           childpid;
        socklen_t                       clilen;
        struct sockaddr_in      cliaddr, servaddr;
        void                            sig_chld(int);

        listenfd = Socket(AF_INET, SOCK_STREAM, 0);

        bzero(&servaddr, sizeof(servaddr));
        servaddr.sin_family      = AF_INET;
        servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
        servaddr.sin_port        = htons(SERV_PORT);

        Bind(listenfd, (SA *) &servaddr, sizeof(servaddr));

        Listen(listenfd, LISTENQ);

        Signal(SIGCHLD, sig_chld); //进行信号注册处理,在子进程结束时候通知父进程。

        for ( ; ; ) {
                clilen = sizeof(cliaddr);
                connfd = Accept(listenfd, (SA *) &cliaddr, &clilen);

                if ( (childpid = Fork()) == 0) {        /* child process */
                        Close(listenfd);        /* close listening socket */
                        str_echo(connfd);       /* process the request */
                        exit(0);
                }
                Close(connfd);                  /* parent closes connected socket */
        }
}

#include        "unp.h"

void sig_chld(int signo)
{
        pid_t   pid;
        int             stat;

        pid = wait(&stat);//阻塞至子进程退出时候继续执行。
        printf("child %d terminated\n", pid);
        return;
}

信号注册行为的函数如下:

/* include signal */
#include        "unp.h"

Sigfunc *signal(int signo, Sigfunc *func)
{
        struct sigaction        act, oact;

        act.sa_handler = func;
        sigemptyset(&act.sa_mask);
        act.sa_flags = 0;
        if (signo == SIGALRM) {
#ifdef  SA_INTERRUPT
                act.sa_flags |= SA_INTERRUPT;   /* SunOS 4.x */
#endif
        } else {
#ifdef  SA_RESTART
                act.sa_flags |= SA_RESTART;             /* SVR4, 44BSD */
#endif
        }
        if (sigaction(signo, &act, &oact) < 0)
                return(SIG_ERR);
        return(oact.sa_handler);
}
/* end signal */

你可能感兴趣的:(Signal)