C++项目实战-信号(二)

简介: C++项目实战-信号(二)

软件条件产生信号

       alarm函数

       说明:设置定时器(闹钟)。在指定多少秒后,内核会给当前进程发送14)SIGALRM信号,进程收到该信号,默认动作终止进程。无论进程处于某种状态都会记时

       原型:

      unsigned int alarm(unsigned int seconds);

       返回0或剩余的秒数

取消定时器:alarm(0)

查看当前进程执行的秒数:time 可执行程序

实际执行时间=系统时间+用户时间+等待时间(绝大多数时间都是在等待中度过)

程序运行的瓶颈在于IO,优化程序,首先优化IO

例如:

        往屏幕上打印,很浪费时间。我们可以重定向往文件中输出

分别向文件和终端打印,观察1s可以计算多少个数:

#include <iostream>
#include <sys/types.h>
#include <unistd.h>
using namespace std;
int main(void)
{
    int i=0;
    alarm(1);   //定时1秒,14)SIGALRM  终止进程
    while(1)
    {
        ++i;
        //cout<<i<<endl;
        printf("%d\n",i);   //printf比cout更快一点点
    }
    return 0;
}

#include <iostream>
#include <sys/types.h>
#include <unistd.h>
#include <sys/stat.h>
#include <fcntl.h>
using namespace std;
int main(void)
{
    int i=0;
    int fd = open("./a.txt",O_RDWR|O_CREAT,0777);
    dup2(fd,STDOUT_FILENO);
    alarm(1);   //定时1秒,14)SIGALRM  终止进程
    while(1)
    {
        ++i;
        //cout<<i<<endl;
        printf("%d\n",i);   //printf比cout更快一点点
    }
    return 0;
}

大概71倍

setitimer函数

说明:设置定时器(闹钟)。可替代alarm函数,精度微妙us,可实现周期定时

原型:int setitimer(int which,const struct itimerval *new_value,struct itimerval *old_value);

成功返回 0 失败-1,并设置error

参数:

       which:指定定时方式

       1.自然定时:ITIMER_REAL  -->  14)SIGARM

       2.虚拟空间计时(用户空间):ITIMER_VIRTUAL  --> 26)SIGVTALRM   占用cpu的时间

       3.运行时间(用户+内存):ITIMER_PROT --> 27)SIGPROF   cpu+系统调用

       old_value

       上一次定时的时间

it_interval:两次触发定时器的间隔时间

it_value:定时器定时的时间

如果都为0,表示清0

案例1:

       使用setitemer函数实现alarm,重复计算1s      

#include <iostream>
#include <sys/time.h>
using namespace std;
void myalarm(unsigned long sec)
{
    itimerval it,oldit;
    it.it_interval.tv_sec = 0;
    it.it_interval.tv_usec = 0;
    it.it_value.tv_sec = 1;
    it.it_value.tv_usec = 0;
    int ret = setitimer(ITIMER_REAL,&it,&oldit);
    if(ret == -1)
    {
        perror("setitimer err");
        exit(-1);
    }
    return;
}
int main(void)
{   
    int i;
    myalarm(1);
    while(1)
    {
        printf("%d\n",i);
        ++i;
    }
    return 0;
}

案例2:

#include <iostream>
#include <sys/time.h>
#include <signal.h>
void func(int signo)
{
    printf("wo shi da sha bi\n");
    //raise(SIGKILL);
}
int main(void)
{
    signal(SIGALRM,func);   //信号捕捉,当SIGALRM递达,则执行func
    itimerval it,oldit;
    it.it_interval.tv_sec = 2;
    it.it_interval.tv_usec = 0;
    it.it_value.tv_sec = 1;
    it.it_value.tv_usec = 0;
    if(setitimer(ITIMER_REAL,&it,&oldit) == -1)
    {
        perror("setitimer err");
        exit(-1);
    }
    while (1)
    {
        /* code */
    }
    return 0;
}

信号集操作函数

       内核通过读取未决信号集来判断信号是否应该被处理。信号屏蔽字mask可以影响未决信号集。而我们可以在应用程序中自定义set来改变mask。以达到屏蔽指定信号的目的。

信号集设定

sigset_t set;                  //unsigned long   (8字节 64bit)

int sigemptyset(sigset_t * set);                                //将某个信号集清0

int sigfillset(sigset_t *set);                                        //将某个信号集置1

int sigaddset(sigset_t *set,int signum);                    //将某个信号加入信号集

int sigdelset(sigset_t *set,int signum);                      //将某个信号清除信号集

int sigismember(const sigset_t *set,int signum);       //判断某个信号是否在信号集中

sigprocmask函数

       int sigprocmask(int how,const sigset_t *set,sigset_t *oldset);

       成功:0  失败:-1,设置error

       参数:

             how:假设当前信号屏蔽字为 mask

               1.SIG_BLOCK:mask = mask | set

               2.SIG_UNBLOCK: mask = mask | ~set

               3.SIG_SETMASK: mask = set

sigpending函数

       读取当前进程的未决信号集

       int sigpending(sigset_t *set);

       成功:0   失败:-1,设置error

 

#include <iostream>
#include <sys/time.h>
#include <signal.h>
#include <unistd.h>
void printset(sigset_t *ped)
{
    for(int i=1;i<32;++i)
    {
        if(sigismember(ped,i) == 1)
        {
            putchar('1');
        }
        else
        {
            putchar('0');
        }
    }
}
int main(void)
{
    sigset_t set,ped;
    sigemptyset(&set);
    sigaddset(&set,SIGINT);
    sigaddset(&set,SIGQUIT);
    sigfillset(&set);
    sigprocmask(SIG_BLOCK,&set,NULL);
    while(1)
    {
        sigpending(&ped);
        printset(&ped);
        sleep(1);
    }
    return 0;
}

 

信号捕捉

signal函数

注册一个信号捕捉函数

       typedef void(*sighandler_t)(int);

       sighandler_t signal(int signum,.sighandler handler);

handler:

       SIG_IGN   忽略此信号

       SIG_DFL   使用信号默认的行为

#include <iostream>
#include <signal.h>
using namespace std;
typedef void (*sighandler_t)(int);
void catchsignal(int signo)
{
    cout<<"aaa"<<endl;
}
int main(void)
{
    sighandler_t headler;
    headler = signal(SIGINT,catchsignal);
    if(headler == SIG_ERR)
    {
        perror("dignal err");
        exit(-1);
    }
    while (1)
    {
        /* code */
    }
    return 0;
}

sigaction函数

      int sigaction(int signum,const struct sigaction *act,struct sigaction *oldact);

       成功:0    失败:-1,并设置error

       act:新的处理方式

       oldact:传出参数,旧的处理方式

       

        重要的参数:

        sa_handler:指定信号捕捉后的处理函数名。(也可以指定SIG_IGN   SIG_DFL)

        sa_mask:调用信号处理函数时,所要屏蔽的信号集合(阻塞信号集)

        sa_flags:通常设置为0,表示默认属性(对正在处理的信息设置屏蔽)

信号捕捉特性

        1.进程正常运行时,默认在PCB中有一个阻塞信号集和未决信号集。对于阻塞信号集,它决定自动屏蔽哪些信号。当注册了某个信号捕捉函数,捕捉到信号以后,要调用该函数。而该函数有可能执行时间很长,在这期间所需要屏蔽的信号由sa_mask来决定。当信号处理完后,又恢复为PCB中的阻塞信号集决定

       2.xxx信号捕捉函数执行期间,xxx信号自动被屏蔽

       3.阻塞的常规信号不支持排队,产生多次只记录一次

信号捕捉函数案例

1.自定义一个信号捕捉函数

#include <iostream>
#include <signal.h>
#include <unistd.h>
using namespace std;
void headler(int signo)
{
    cout<<"我是你大爷..."<<endl;
    sleep(2);
    return;
}
int main(void)
{
    struct sigaction act;
    act.sa_handler = headler;
    act.sa_flags = 0;
    sigemptyset(&act.sa_mask);
    sigaddset(&act.sa_mask,SIGQUIT);
    int ret = sigaction(SIGINT,&act,NULL);
    if(ret == -1)
    {
        perror("sigaction err");
        exit(-1);
    }
    while(1)
    {
    }
    return 0;
}

2.验证在信号处理函数执行期间,该信号多次递送,那么只在处理函数支行技术和,处理一次

#include <iostream>
#include <signal.h>
#include <unistd.h>
using namespace std;
void headler(int signo)
{
    cout<<"我是你大爷..."<<endl;
    sleep(2);
    cout<<"-----end-------"<<endl;
    return;
}
int main(void)
{
    struct sigaction act,old;
    act.sa_handler = headler;
    act.sa_flags = 0;
    sigemptyset(&act.sa_mask);
    sigaction(SIGINT,&act,&old);
    while(1)
    {
    }
    sigaction(SIGINT,&old,NULL);
    return 0;
}

3.验证sa_mask的屏蔽作用

#include <iostream>
#include <signal.h>
#include <unistd.h>
using namespace std;
void headler(int signo)
{
    cout<<"我是你大爷..."<<endl;
    sleep(2);
    return;
}
int main(void)
{
    struct sigaction act;
    act.sa_handler = headler;
    act.sa_flags = 0;
    sigemptyset(&act.sa_mask);
    sigaddset(&act.sa_mask,SIGQUIT);
    int ret = sigaction(SIGINT,&act,NULL);
    if(ret == -1)
    {
        perror("sigaction err");
        exit(-1);
    }
    while(1)
    {
    }
    return 0;
}

内核实现信号捕捉过程

信号的产生和处理都是由内核进行的

频繁的在用户态和内核态进行切换是很浪费时间的

时态竞争

       当前进程在执行期间,由于竞争的原因,导致程序执行的时序在先后两次的执行中有不同的结果

       比如A准备睡觉,定了一个闹钟(定时器),在10s后叫醒他。

       由于时序可能会有两种情况:

       正常:10s后A被闹钟叫醒

       异常:A在5s时被B叫醒,并且A跟着B一起去打牌,打了20s,那么在打牌的期间,

               闹钟正常响起,但是并没有叫醒A

       

时序问题分析:

       假设我们想要通过pause和alarm实现sleep:

       1.注册SIGALARM信号处理函数(sigaction...)

       2.调用alarm(1)函数设定闹钟为1秒

       3.函数调用结束,开始计时倒数。正在这时,进程失去CPU(时间片轮转,进程调度算法决定),该进程处于就绪状态,等待CPU

       4.但是此时时自然计时,倒数依旧在进行着,如果此时倒数结束,alarm发送信号,信号将递达,但是此时进程处于挂起状态,每一办法通过信号捕捉函数进行处理,所有处于未决状态。

       5.当进程再次获得CPU,SIGALARM信号递达,执行处理函数函数

       6.信号处理函数结束,回到主控制程序,执行pause(),进程将会被挂起,等待alarm唤醒,可以再没有人来唤醒他了

#include <iostream>
using namespace std;
void myheadler(int signo)
{
    cout<<"aa"<<endl;
}
int main(void)
{
    signal(SIGALARM,myheadler);
    alarm(1);
    pause();
    while(1);
    return 0;
}

解决时序问题

       可以通过设置屏蔽SIGALRM的方法来控制程序执行逻辑,但无论如何设置,程序都有可能在“解除信号屏蔽字”与“挂起等待信号”这个两个操作间隙失去CPU。除非将这两个步骤合成原子操作。sigsuspend可以实现。

      int sigsuspend(const sigset_t *mask)   挂起等待信号

      sigsuspend函数调用期间,进程信号屏蔽字由参数mask决定

      可将某个信号从临时信号屏蔽集种删除,这样在调用sigsuspend时将解除对该信号的屏蔽,然后挂起等待,当sigsuspend返回时,信号屏蔽字恢复原来的值。

#include <iostream>
using namespace std;
void myheadler(int signo)
{
    cout<<"aa"<<endl;
}
int main(void)
{
    signal(SIGALARM,myheadler);
    sigset_t mask;
    sigfillset(&mask);
    sigdelset(&mask,SIGALARM);
    alarm(1);
    sigsuspend(&mask);
    while(1);
    return 0;
}

时序问题总结

       竞态条件,跟系统负载有很紧密的关系,体现除信号的不可靠性。系统负载越严重,信号不可靠性越强。不可靠由其实现原理所致。信号是通过软件方式实现,每次系统调用结束后或中断处理结束后,需通过扫描PCB中的未决信号集来判断是否应处理某个信号。当系统负载过重时,会出现混乱

可不可重入函数

一个函数在被调用执行期间(尚未调用结束),由于某时序又被重复调用,称之为"重入"。

1.定义可重入函数,函数内不能包含全局变量集static变量,不能使用malloc、free

2.信号捕捉函数应设计为可重入函数

3.不可重入的原理:使用静态数据结构    调用了malloc和new (不是栈结构)  是标准I/O

SIGCHLD信号

       当子进程停止或者结束时,会向父进程发送SIGCHLD信号,该信号默认处理动作是忽略,所有产生僵尸进程。

       我们之前的程序是,在父进程中调用 wait或者waitpid来回收子进程,此时父进程要么阻塞等待,要么非阻塞而采用轮询的方式。那么将会导致父进程不能做其他工作,只能等着回收子进程。

       我们可以利用SIGCHLD信号来捕捉,当有SIGCHLD信号产生时,父进程执行信号捕捉函数。

       我们先来看一段程序:

       

#include <iostream>
#include <signal.h>
#include <unistd.h>
#include <wait.h>
using namespace std;
void sigchldHeadler(int signo)
{
    pid_t pid;
    int status;
    while((pid=waitpid(-1,&status,WNOHANG))>0)
    {
        printf("回收成功:%d   ok\n",pid);
        if(WIFEXITED(status))
        {
            printf("退出状态:%d\n",WEXITSTATUS(status));
        }
        if(WIFSIGNALED(status))
        {
            printf("退出状态(信号):%d\n",WTERMSIG(status));
        }
    }
}
int main(void)
{
    //创建10个子进程
    int i;
    for(i=0;i<10;i++)
    {
        pid_t pid = fork();
        if(pid == 0)
        {
            break;
        }
    }
    if(i < 10)
    {
        printf("I am child:%d\n",getpid());
        sleep(1);
    }
    else if(i == 10)
    {      
        struct sigaction act;
        act.sa_handler = sigchldHeadler;
        act.sa_flags = 0;
        sigemptyset(&act.sa_mask);
        sigaction(SIGCHLD,&act,NULL);
        while(1)
        {
            printf("I am parent:%d\n",getpid());   
            sleep(1);
        }
    }
    return 0;
}

我们来思考几个问题:

当父进程在执行信号捕捉函数时,又有子进程死亡。或者有多个子进程死亡。我们知道信号集是位图机制,是不支持排队的。

当父进程的信号捕捉函数还没有注册,就已经有子进程结束了。导致僵尸进程...

或者在fork之前将SIGCHLD信号设置屏蔽

完整:

#include <iostream>
#include <signal.h>
#include <unistd.h>
#include <wait.h>
using namespace std;
void sigchldHeadler(int signo)
{
    pid_t pid;
    int status;
    while((pid=waitpid(-1,&status,WNOHANG))>0)
    {
        printf("回收成功:%d   ok\n",pid);
        if(WIFEXITED(status))
        {
            printf("退出状态:%d\n",WEXITSTATUS(status));
        }
        if(WIFSIGNALED(status))
        {
            printf("退出状态(信号):%d\n",WTERMSIG(status));
        }
    }
}
int main(void)
{
    //设置阻塞
    sigset_t set;
    sigemptyset(&set);
    sigdelset(&set,SIGCHLD);
    sigprocmask(SIG_BLOCK,&set,NULL);
    //创建10个子进程
    int i;
    for(i=0;i<10;i++)
    {
        pid_t pid = fork();
        if(pid == 0)
        {
            break;
        }
    }
    if(i < 10)
    {
        printf("I am child:%d\n",getpid());
        //sleep(1);
    }
    else if(i == 10)
    {      
        struct sigaction act;
        act.sa_handler = sigchldHeadler;
        act.sa_flags = 0;
        sigemptyset(&act.sa_mask);
        sigaction(SIGCHLD,&act,NULL);
        while(1)
        {
            printf("I am parent:%d\n",getpid());   
            sleep(1);
        }
    }
    return 0;
}

信号传参

前面讲到信号是不能携带大量数据的,一般通过 kill 来发送信号

但是信号可以携带数据,可以携带少量。

通过sigqueue函数,可在向指定进程发送信号的同时携带参数

int sigqueue(pid_t pid,int sig,const union sigval value);

成功返回0,失败-1,设置error

注意事项:

       向指定进程发送指定信号的同时携带数据。不能够传地址,不能进程之间虚拟地址空间各自独立,当前进程地址传递给另一进程没有实际意义

捕捉函数:

       

  int sigaction(int signum,const struct sigaction *act,struct sigaction *oldact);

       成功:0    失败:-1,并设置error

       act:新的处理方式

       oldact:传出参数,旧的处理方式

       

       

       不使用sa_handler,而使用sa_sigaction。sa_flags必须指定为SA_SIGINFO

       

中断系统调用

系统调用分为两类:

慢速系统调用:可能导致进程永远阻塞,如果在阻塞期收到一个信号,该系统调用就会被断开,不再继续执行,也可以设定系统调用重启(read、write、pause、wait...)

其他系统调用:getpid,getpid、fork

sa_flags可以设置为 SA_RESTART

相关文章
|
网络协议 C++ iOS开发
C++项目实战-UDP服务器
C++项目实战-UDP服务器
129 0
|
存储 网络协议 Linux
C++项目实战-实际应用(一)
C++项目实战-实际应用(一)
106 0
|
3月前
|
Unix Linux C++
Linuxc/c++之信号基础
这篇文章详细介绍了Linux下C/C++信号的基本概念、产生原因、处理过程、分类、注册与发送方法,以及信号屏蔽的机制。
25 0
Linuxc/c++之信号基础
|
8月前
|
算法 数据处理 C++
Franca IDL与CommonAPI C++ D-Bus实战教程:方法、信号和属性传递详解
Franca IDL与CommonAPI C++ D-Bus实战教程:方法、信号和属性传递详解
163 0
|
存储 网络协议 Unix
C++项目实战-socket编程
C++项目实战-socket编程
98 1
|
8月前
|
算法 编译器
[C++&Qt] 通过信号与槽传递数据
[C++&Qt] 通过信号与槽传递数据
200 0
|
存储 NoSQL Linux
C++信号的使用
C++信号的使用
118 0
|
缓存 监控 Linux
C++项目实战-高并发服务器详析(二)
C++项目实战-高并发服务器详析(二)
87 0
|
Linux 调度 C++
C++项目实战-高并发服务器详析(一)
C++项目实战-高并发服务器详析(一)
240 0
|
Java 调度 C++
C++项目实战--线程池代码讲解
C++项目实战--线程池代码讲解
104 0