您的位置:首页 > 其它

进程同步(三)—— 信号

2016-05-02 13:40 357 查看

程序在执行过程经常产生信号,有些是由内核发出,有些是由用户发出。

执行命令kill -l可以查看信号。







这些信号由系统定义,他们不是简单的int之类的数据类型,可能是调用一些函数。

这些信号中,10/12是给用户预留使用的,其他信号各有自己相应的含义。代码中无法完美的自定义信号,容易覆盖或产生系统错误,所以还是乖乖用系统提供的信号吧。

信号类似QT的信号槽的关系,一个信号对应一个处理函数,可以对相应的信号进行屏蔽之类的处理。



信号处理函数属于可重入函数,可以随时中断,允许多个副本执行。

信号处理属于异步处理,和主程是分开执行的,内核发送信号只发送一次。一般处理函数仅仅用于通知主程收到信号,有主程处理具体内容。



下面是信号处理代码,分别是内核发送的子进程结束信号和用户自定义信号。

#include <sys/types.h>
#include <unistd.h>
#include <sys/wait.h>
#include <signal.h>
#include <string.h>
#include <iostream>
#include <stdlib.h>

using namespace std;

class sigOp
{
public:
void addSigProcess(int sig,void (*func)(int));
void sendSig(const int sig, const int pid);
};
void sigOp::addSigProcess(int sig,void (*func)(int))
{
struct sigaction stuSig;
memset(&stuSig, '\0', sizeof(stuSig));
stuSig.sa_handler = func;
stuSig.sa_flags |= SA_RESTART;
sigfillset(&stuSig.sa_mask);
sigaction(sig, &stuSig, NULL);
}
void sigOp::sendSig(const int sig, const int pid)
{
kill(pid, sig);
cout<<"send!"<<endl;
}
void recSig(int sig)
{
cout<<"rev sig!"<<endl;
}
void waitchlid(int sig)
{
pid_t pid;
int stat;
while((pid = waitpid(-1, &stat, WNOHANG)) > 0);
}
int main()
{
sigOp sig;
sig.addSigProcess(SIGUSR1, recSig);
sig.addSigProcess(SIGCHLD, waitchlid);
pid_t pid = fork();
if (pid > 0)
{
sig.sendSig(SIGUSR1, pid);
}
else
{
return 0;
}
while(1);
return 0;
}




当我们不处理SIGCHLD信号时,利用ps -aux查看,可以看到进程属于僵尸态。





收到SIGCHLD信号,并且waitpid回收后,没有僵尸进程。

测试结果:









内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: