相关函数
signal,sigprocmask,sigpending,sigsuspend
表头文件
#include
定义函数
int sigaction(int signum,const struct sigaction *act ,struct sigaction *oldact);
函数说明
sigaction()会依参数signum指定的信号编号来设置该信号的处理函数。参数signum可以指定SIGKILL和SIGSTOP以外的所有信号。
如参数结构sigaction定义如下
struct sigaction
{
void (*sa_handler) (int);
sigset_t sa_mask;
int sa_flags;
void (*sa_restorer) (void);
}
sa_handler此参数和signal()的参数handler相同,代表新的信号处理函数,其他意义请参考signal()。
sa_mask 用来设置在处理该信号时暂时将sa_mask 指定的信号搁置。
sa_restorer 此参数没有使用。
sa_flags 用来设置信号处理的其他相关操作,下列的数值可用。
OR 运算(|)组合
A_NOCLDSTOP : 如果参数signum为SIGCHLD,则当子进程暂停时并不会通知父进程
SA_ONESHOT/SA_RESETHAND:当调用新的信号处理函数前,将此信号处理方式改为系统预设的方式。
SA_RESTART:被信号中断的系统调用会自行重启
SA_NOMASK/SA_NODEFER:在处理此信号未结束前不理会此信号的再次到来。
如果参数oldact不是NULL指针,则原来的信号处理方式会由此结构sigaction 返回。
返回值
执行成功则返回0,如果有错误则返回-1。
错误代码
EINVAL 参数signum 不合法, 或是企图拦截SIGKILL/SIGSTOPSIGKILL信号
EFAULT 参数act,oldact指针地址无法存取。
EINTR 此调用被中断
范例
#include
#include
void show_handler(struct sigaction * act)
{
switch (act->sa_flags)
{
case SIG_DFL:printf(“Default actionn”);break;
case SIG_IGN:printf(“Ignore the signaln”);break;
default: printf(“0x%xn”,act->sa_handler);
}
}
main()
{
int i;
struct sigaction act,oldact;
act.sa_handler = show_handler;
act.sa_flags = SA_ONESHOT|SA_NOMASK;
sigaction(SIGUSR1,&act,&oldact);
for(i=5;i<15;i++)
{
printf(“sa_handler of signal %2d =”.i);
sigaction(i,NULL,&oldact);
}
}
执行
sa_handler of signal 5 = Default action
sa_handler of signal 6= Default action
sa_handler of signal 7 = Default action
sa_handler of signal 8 = Default action
sa_handler of signal 9 = Default action
sa_handler of signal 10 = 0x8048400
sa_handler of signal 11 = Default action
sa_handler of signal 12 = Default action
sa_handler of signal 13 = Default action
sa_handler of signal 14 = Default action
    sigaddset(增加一个信号至信号集)
    
    相关函数
    
    sigemptyset,sigfillset,sigdelset,sigismember
    
    表头文件
    
    #include
    
    定义函数
    
    int sigaddset(sigset_t *set,int signum);
    
    函数说明
    
    sigaddset()用来将参数signum 代表的信号加入至参数set 信号集里。
    
    返回值
    
    执行成功则返回0,如果有错误则返回-1。
    
    错误代码
    
    EFAULT 参数set指针地址无法存取
    
    EINVAL 参数signum非合法的信号编号
    
   
    sigdelset(从信号集里删除一个信号)
    
    相关函数
    
    sigemptyset,sigfillset,sigaddset,sigismember
    
    表头文件
    
    #include
    
    定义函数
    
    int sigdelset(sigset_t * set,int signum);
    
    函数说明
    
    sigdelset()用来将参数signum代表的信号从参数set信号集里删除。
    
    返回值
    
    执行成功则返回0,如果有错误则返回-1。
    
    错误代码
    
    EFAULT 参数set指针地址无法存取
    
    EINVAL 参数signum非合法的信号编号
    
   
    sigemptyset(初始化信号集)
    
    相关函数
    
    sigaddset,sigfillset,sigdelset,sigismember
    
    表头文件
    
    #include
    
    定义函数
    
    int sigemptyset(sigset_t *set);
    
    函数说明
    
    sigemptyset()用来将参数set信号集初始化并清空。
    
    返回值
    
    执行成功则返回0,如果有错误则返回-1。
    
    错误代码
    
    EFAULT 参数set指针地址无法存取
    
   
    sigfillset(将所有信号加入至信号集)
    
    相关函数
    
    sigempty,sigaddset,sigdelset,sigismember
    
    表头文件
    
    #include
    
    定义函数
    
    int sigfillset(sigset_t * set);
    
    函数说明
    
    sigfillset()用来将参数set信号集初始化,然后把所有的信号加入到此信号集里。
    
    返回值
    
    执行成功则返回0,如果有错误则返回-1。
    
    附加说明
    
    EFAULT 参数set指针地址无法存取
    
   
    sigismember(测试某个信号是否已加入至信号集里)
    
    相关函数
    
    sigemptyset,sigfillset,sigaddset,sigdelset
    
    表头文件
    
    #include
    
    定义函数
    
    int sigismember(const sigset_t *set,int signum);
    
    函数说明
    
    sigismember()用来测试参数signum 代表的信号是否已加入至参数set信号集里。如果信号集里已有该信号则返回1,否则返回0。
    
    返回值
    
    信号集已有该信号则返回1,没有则返回0。如果有错误则返回-1。
    
    错误代码
    
    EFAULT 参数set指针地址无法存取
    
    EINVAL 参数signum 非合法的信号编号
    
   
    signal(设置信号处理方式)
    
    相关函数
    
    sigaction,kill,raise
    
    表头文件
    
    #include
    
    定义函数
    
    void (*signal(int signum,void(* handler)(int)))(int);
    
    函数说明
    
    signal()会依参数signum 指定的信号编号来设置该信号的处理函数。当指定的信号到达时就会跳转到参数handler指定的函数执行。如果参数handler不是函数指针,则必须是下列两个常数之一:
    
    SIG_IGN 忽略参数signum指定的信号。
    
    SIG_DFL 将参数signum 指定的信号重设为核心预设的信号处理方式。
    
    关于信号的编号和说明,请参考附录D
    
    返回值
    
    返回先前的信号处理函数指针,如果有错误则返回SIG_ERR(-1)。
    
    附加说明
    
    在信号发生跳转到自定的handler处理函数执行后,系统会自动将此处理函数换回原来系统预设的处理方式,如果要改变此操作请改用sigaction()。
    
    范例
    
    参考alarm()或raise()。
    
   
    sigpending(查询被搁置的信号)
    
    相关函数
    
    signal,sigaction,sigprocmask,sigsuspend
    
    表头文件
    
    #include
    
    定义函数
    
    int sigpending(sigset_t *set);
    
    函数说明
    
    sigpending()会将被搁置的信号集合由参数set指针返回。
    
    返回值执
    
    行成功则返回0,如果有错误则返回-1。
    
    错误代码
    
    EFAULT 参数set指针地址无法存取
    
    EINTR 此调用被中断。
    
   
    sigprocmask(查询或设置信号遮罩)
    
    相关函数
    
    signal,sigaction,sigpending,sigsuspend
    
    表头文件
    
    #include
    
    定义函数
    
    int sigprocmask(int how,const sigset_t *set,sigset_t * oldset);
    
    函数说明
    
    sigprocmask()可以用来改变目前的信号遮罩,其操作依参数how来决定
    
    SIG_BLOCK 新的信号遮罩由目前的信号遮罩和参数set 指定的信号遮罩作联集
    
    SIG_UNBLOCK 将目前的信号遮罩删除掉参数set指定的信号遮罩
    
    SIG_SETMASK 将目前的信号遮罩设成参数set指定的信号遮罩。
    
    如果参数oldset不是NULL指针,那么目前的信号遮罩会由此指针返回。
    
    返回值
    
    执行成功则返回0,如果有错误则返回-1。
    
    错误代码
    
    EFAULT 参数set,oldset指针地址无法存取。
    
    EINTR 此调用被中断
   
。。。。。。。。。。。。。。。例程。。。。。。。。。。。。。。。。。。。。
======================================
    #include <stdio.h>
    
    #include <unistd.h>
    
    #include <signal.h>
    
    #include <stdlib.h>
   
    
    static void sig_quit(int signo)
    
    {
    
    
    printf(“caught SIGQUIT/n”);
    
    signal(SIGQUIT, SIG_DFL);
    
    }
   
    int main()
    
    {
    
    
    sigset_t newmask;
    
    sigset_t oldmask;
    
    sigset_t pendmask;
   
signal(SIGQUIT, sig_quit);
    sigemptyset(&newmask);
    
    sigaddset(&newmask, SIGQUIT);
   
sigprocmask(SIG_BLOCK, &newmask, &oldmask);
sleep (5);
sigpending(&pendmask);
    //        sleep (5);
    
    if (sigismember(&pendmask, SIGQUIT))
    
    {
    
    
    printf(“/nSIGQUIT pending/n”);
    
    }
   
sigprocmask(SIG_SETMASK, &oldmask, NULL);
printf(“SIGQUIT unblocked/n”);
sleep(5);
return (0);
    }
    
   
============================================
    #include <time.h>
    
    #include <stdio.h>
    
    #include <signal.h>
    
    #include <sys/types.h>
    
    #include <errno.h>
    
    #include <stdlib.h>
    
    #include <unistd.h>
    int main(int argc, char *argv[])
    
    {
    
    sigset_t allSignal;
    
    sigfillset(&allSignal);
    
    if (-1 == sigprocmask(SIG_SETMASK, &allSignal, NULL))
    
    {
    
    
    exit(-1);
    
    }
    
    alarm(2);
   
    
    sigset_t waitSignals;
    
    sigemptyset(&waitSignals);
    
    sigaddset(&waitSignals, SIGTERM);
    
    sigaddset(&waitSignals, SIGALRM);
    
    sigaddset(&waitSignals, SIGHUP);
   
    
    int pid = fork();
   
    if(pid<0)
    
    {
    
    
    printf(“fork error!/n”);
    
    exit(1);
    
    }
    
    else if (pid == 0)
    
    {
    
    
    printf(“child pid:%d/n”,getpid());
    
    printf(“child parent pid:%d/n”,getppid());
   
    for(int i = 0; i < 10; i++)
    
    {
    
    
    sleep(1);
    
    if((kill(getppid(),SIGHUP))==-1)
    
    {
    
    
    perror(“kill”);
    
    exit(1);
    
    }
}
    kill(getppid(),SIGTERM);
    
    exit(EXIT_SUCCESS);
    
    }
   
    if(pid>0)
    
    {
    
    printf(“parent pid:%d/n”,getpid());
    
    while (1)
    
    {
    
    
    int sign = 0;
   
    if (sigwait(&waitSignals, &sign) != 0)
    
    {
    
    
    continue;
    
    }
    
    if (SIGTERM == sign)
    
    {
    
    
    printf(“signal SIGTERM/n”);
    
    exit(0);
    
    }
    
   
    if (SIGHUP == sign)
    
    {
    
    
    printf(“signal SIGHUP/n”);
   
}
    if (SIGALRM == sign)
    
    {
    
    
    printf(“signal SIGALRM/n”);
    
    alarm(2);
    
    }
   
    }
    
    }
   
    }
    
   
 
