函数简介篇——信号集相关示例

说明
  本文章旨在总结备份、方便以后查询,由于是个人总结,如有不对,欢迎指正;另外,内容大部分来自网络、书籍、和各类手册,如若侵权请告知,马上删帖致歉。
  QQ 群 号:513683159 【相互学习】
内容来源
  《Unix环境高级编程》

示例一:

1️⃣说明:

  开始5秒,无论多少次Ctrl+\,进程将被阻塞并将处理方式改为默认方式,后再输出字符串。
  最后5秒,由于已修改为默认方式故直接终止进程。

2️⃣源文件:sigmask.c
#include <stdio.h>
#include <unistd.h>
#include <signal.h>

static void sig_quit(int signo)
{
    printf("caught SIGQUIT\n");
    if(signal(SIGQUIT,SIG_DFL) == SIG_ERR)
        fprintf(stderr,"can't reset SIGQUIT");
}

int main(int argc, char *argv[])
{
    sigset_t newmask, oldmask, pendmask;

    /* 
        前提:SIGQUIT = 终端退出信号(Ctrl+\),终止进程组并产生core文件 
        若进程捕捉到SIGQUIT时,调用sig_quit回调函数输出:caught SIGQUIT
    */
    if (signal(SIGQUIT, sig_quit) == SIG_ERR)
        fprintf(stderr, "can't catch SIGQUIT!");

    /*
         阻塞SIGQUIT并保存当前信号屏蔽字
    */
    sigemptyset(&newmask);                                  //清空信号集
    sigaddset(&newmask, SIGQUIT);                           //信号集中加入SIGQUIT
    if (sigprocmask(SIG_BLOCK, &newmask, &oldmask) < 0)     //新信号集是当前信号集和newmask的并集,包含希望阻塞的附加信号,并通过oldmask返回信号集
        fprintf(stderr, "SIGBLOCK error");

    /* 休眠5秒,在此期间产生的SIGQUIT信号会被阻塞 */
    sleep(5);

    /* 检查信号是否是未决的,将其设置为不再阻塞 */
    if (sigpending(&pendmask) < 0)                           //取出pending集状态
        fprintf(stderr, "sigpending error");
    if (sigismember(&pendmask, SIGQUIT))                     //判断信号集中是否含有SIGQUIT
        fprintf(stdout, "\nSIGQUIT pending\n");

    /* 复位信号掩码,解除SIGQUIT阻塞 */
    if (sigprocmask(SIG_SETMASK, &oldmask, NULL) < 0)        //新的信号集复位为oldmask
        fprintf(stderr, "SIG_SETMASK error\n");
    fprintf(stdout, "SIGQUIT unblocked\n");

    /*休眠5秒,若再次产生退出信号,则会因上次捕捉到该信号时,已将处理方式设置为默认动作,故此次会终止 */
    sleep(5);

    return 0;
}
编译运行及结果
$ gcc sigmask.c -o sigmask
$ ./sigmask
^\^\^\^\^\^\^\^\^\^\^\					无论多少次Ctrl+\(5秒内)
SIGQUIT pending						    从sleep返回后
caught SIGQUIT							在信号处理程序中
SIGQUIT unblocked					    在sigprocmask返回后
^\Quit (core dumped)					一次Ctrl+\直接退出

示例二:

  程序效果描述:一直打印5个*换行,若Ctrl+C进行中断,则无法中断,也不会阻塞,而是在下一行开始时进行响应。(多个信号只会响应一次),此时若想退出可使用Ctrl+\

#include <stdio.h>
#include <signal.h>
#include <unistd.h>


static void int_handler(int s)
{
        write(1,"!",1);
}

int main(int argc,char *argv[])
{
        int i;
        sigset_t set,oset,saveset;                            //声明信号集合

        signal(SIGINT,int_handler);                   //若程序结束前,收到SIGINT信号时执行int_handler该行为
        sigemptyset(&set);							  //将信号集清空
        sigaddset(&set,SIGINT);              		  //将SIGINT加入信号集中    
        sigprocmask(SIG_UNBLOCK,&set,&saveset); 	  //用于恢复该模块使用之前的信号集
        while(1)
        {
              
                sigprocmask(SIG_BLOCK,&set,&oset);    //将对该set信号集进行阻塞,并将之前信号读入oset中
                for(i = 0 ; i < 5; i++)
                {
                        write(1,"*",1);
                        sleep(1);
                }
                write(1,"\n",1);
                sigprocmask(SIG_UNBLOCK,&set,NULL);   //恢复信号集不阻塞
                //sigprocmask(SIG_SETMASK,&oset,NULL); 恢复信号集为原来的样子
        }
         sigprocmask(SIG_SETMASK,&saveset,NULL); 	  //用于恢复该模块使用之前的信号集
        return 0;
}

示例三:(sigsuspend()示例)

  程序效果描述:打印一行5个*换行后阻塞,若Ctrl+C发送信号,则在下一行开始时进行响应。(多个信号只会响应一次),此时若想退出可使用Ctrl+\

#include <stdio.h>
#include <signal.h>
#include <unistd.h>


static void int_handler(int s)
{
        write(1,"!",1);
}

int main(int argc,char *argv[])
{
        int i;
        sigset_t set,oset,saveset;                            //声明信号集合

        signal(SIGINT,int_handler);                           //若程序结束前,收到SIGINT信号时执行int_handler该行为
        sigemptyset(&set);                                                        //将信号集清空
        sigaddset(&set,SIGINT);                           //将SIGINT加入信号集中    
        sigprocmask(SIG_UNBLOCK,&set,&saveset);           //解除set的阻塞,saveset恢复该模块使用之前的信号集
        sigprocmask(SIG_BLOCK,&set,&oset);                //阻塞set ,
        while(1)
        {
                for(i = 0 ; i < 5; i++)
                {
                        write(1,"*",1);
                        sleep(1);
                }
                write(1,"\n",1);
                sigsuspend(&oset);                      //相当于下面三句的原子操作
                /*
                sigset_t tmpset; 
                sigprocmask(SIG_SETMASK,&oset,&tmpset); 
                pause();
                sigprocmask(SIG_SETMASK,&tmpset,NULL);
                */
        }
         sigprocmask(SIG_SETMASK,&saveset,NULL);          //用于恢复该模块使用之前的信号集
        return 0;
}

示例四:(sigaction()示例)

  原:守护进程程序

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <syslog.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>

#define FNAME "/tmp/out"

static int daemonize(void)
{
        pid_t pid;
        int fd;

        pid = fork();
        if(pid < 0)
                return -1;

        if(pid > 0)     //parent
                exit(0);

        fd = open("/dev/null",O_RDWR);
        if(fd < 0)
                return -1;

        dup2(fd,0);
        dup2(fd,1);
        dup2(fd,2);
        if(fd > 2)
                close(fd);

        setsid();

        chdir("/");
        //umask(0);
        return 0;
}

int main()
{
        int i;
        FILE *fp;

        openlog("mydaemon",LOG_PID,LOG_DAEMON);

        if(daemonize())
        {
                syslog(LOG_ERR,"daemonize() failed!");
                exit(1);
        }
        else
        {
                syslog(LOG_INFO,"daemonize() successded!!");
        }

        fp = fopen(FNAME,"w");
        if(fp == NULL)
        {
                syslog(LOG_ERR,"fopen():%s",strerror(errno));
                exit(1);
        }

        syslog(LOG_INFO,"%s was opened.",FNAME);

        for(i = 0;;i++)
        {
                fprintf(fp,"%d\n",i);
                fflush(fp);
                syslog(LOG_DEBUG,"%d is printef.",i);
                sleep(1);
        }

        fclose(fp);
        closelog();

        exit(0);
}


  修改后守护进程程序

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <syslog.h>
#include <unistd.h>
#include <string.h>
#include <errno.h>
#include <signal.h>

static FILE *fp;
#define FNAME "/tmp/out"

static int daemonize(void)
{
        pid_t pid;
        int fd;

        pid = fork();
        if(pid < 0)
                return -1;

        if(pid > 0)     //parent
                exit(0);

        fd = open("/dev/null",O_RDWR);
        if(fd < 0)
                return -1;

        dup2(fd,0);
        dup2(fd,1);
        dup2(fd,2);
        if(fd > 2)
                close(fd);

        setsid();

        chdir("/");
        //umask(0);
        return 0;
}

static void daemon_exit(int s)
{

        fclose(fp);
        closelog();
        exit(0);
}

int main()
{
        int i;
        struct sigaction sa;
        sa.sa_handler = daemon_exit;
        sigemptyset(&sa.sa_mask);
        sigaddset(&sa.sa_mask,SIGQUIT);
        sigaddset(&sa.sa_mask,SIGTERM);
        sigaddset(&sa.sa_mask,SIGINT);
        sa.sa_flags = 0;
        sigaction(SIGINT,&sa,NULL);
        sigaction(SIGQUIT,&sa,NULL);
        sigaction(SIGTERM,&sa,NULL);
        //signal(SIGINT,daemon_exit);
        //signal(SIGQUIT,daemon_exit);
        //signal(SIGTERM,daemon_exit);
        openlog("mydaemon",LOG_PID,LOG_DAEMON);

        if(daemonize())
        {
                syslog(LOG_ERR,"daemonize() failed!");
                exit(1);
        }
        else
        {
                syslog(LOG_INFO,"daemonize() successded!!");
        }

        fp = fopen(FNAME,"w");
        if(fp == NULL)
        {
                syslog(LOG_ERR,"fopen():%s",strerror(errno));
                exit(1);
        }

        syslog(LOG_INFO,"%s was opened.",FNAME);

        for(i = 0;;i++)
        {
                fprintf(fp,"%d\n",i);
                fflush(fp);
                syslog(LOG_DEBUG,"%d is printef.",i);
                sleep(1);
        }


        exit(0);
}




  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值