1.异步事件通信(甚至可以附带信息数据)
(额,由于无线网卡掉了,电脑切换到ubuntu的情况下无法联网,暂时不自己写一个了)暂且用文字说明:
其实子进程结束,内核产生SIGCHLD并递送给父进程,就是一个典型的异步事件通信
2.保护临界区不被信号中断
以下摘自unix高级环境编程
#include "apue.h"
static void sig_int(int);
int
main(void)
{
sigset_t newmask, oldmask, waitmask;
pr_mask("program start: ");
if(signal(SIGINT, sig_int) == SIG_ERR)
err_sys("signal(SIGINT) error");
sigemptyset(&waitmask);
sigaddset(&waitmask, SIGUSR1);
sigemptyset(&newmask);
sigaddset(&newmask, SIGINT);
/*
* Block SIGINT and save current signal mask.
*/
if(sigprocmask(SIG_BLOCK, &newmask, &oldmask) < 0) //先设置要阻塞的信号
err_sys("SIG_BLOCK error: ");
/*
* Critical region of code.
*/
pr_mask("in critical region: ");
/*
* Pause, allowing all signals except SIGUSR1.
*/
if(sigsuspend(&waitmask) != -1) //设置信号集(一般就不包括之前设置的阻塞信号,从而使之前产生的信号解阻塞,则调用信号处理函数)
err_sys("sigsuspend error");
pr_mask("after return from sigsuspend: ");
/*
* Reset signal mask which unblocks SIGINT.
*/
if(sigprocmask(SIG_SETMASK, &oldmask, NULL) < 0) //还原屏蔽信号字
err_sys("SIG_SETMASK error");
/*
* And continue processing...
*/
pr_mask("program exit: ");
exit(0);
}
static void
sig_int(int signo)
{
pr_mask("\nin sig_int: ");
}
3.等待一个信号处理程序设置一个全局变量
略,实在不喜欢使用全局变量
4.实现父子进程间的同步
- 假设有两个进程,一个进程需要等待某些数据的到位才能接下来的处理,此时先设置好用户的一个自定义信号(如SIG_USR1)的捕捉函数(此时我们需要额外的数据信息,要采用sigaction中的最后一个参数),然后调用sigsuspend设置进程的信号屏蔽字并阻塞(原子操作,一气呵成),等待制定信号和附带数据的到来)。另一个进程在计算之后得到数据结果,通过sigqueue函数传递给等待进程SIG_USR1信号和附带数据(计算结果)。
- 还有一种特例,父进程需要等待子进程的最终计算结果才能继续执行的话,可以调用wait得到子进程的时机,那么我们如何在父子进程之间传递数据信息呢,此时就需要我们熟悉进程间通信(ipc)这个问题了,参见unix网络编程ipc