屏蔽信号的多路选择I/O

前边提到了多路I/O的方法,这一章屏蔽信号的多路选择与之前的多路I/O一致,只是增加了屏蔽信号的作用。多路选择I/O中我们使用的是select函数,屏蔽信号的多路选择I/O使用的是pselect函数,与之前的函数相比,增加了一个参数可以用来屏蔽信号。具体函数如下所示:

int pselect(int maxfdp1, fd_set *readset, fd_set *writeset, fd_set *exceptset, const struct timespec *timeout, const sigset_t *sigmask);

头文件: #include <sys/select.h>

参数说明:前4个参数与select函数的参数相同,分别表示最大的文件描述符和关心的文件中状态。

第5个参数表示等待时间,所不同的是timespec结构所能表示的最小精度是纳秒,旧的结构体中所能表示的最小精度是微妙数。

pselect函数最后一个参数可以用来屏蔽信号,在pselect函数返回后,再将屏蔽的信号恢复,并且所有的操作都是原子的。

返回值:超时返回0,出错返回-1,成功返回描述字的个数。

下面用两个程序的对比来说明select函数与pselect函数的区别。

第一个程序使用select函数,程序中人为的制造了一个造成函数阻塞的条件。该程序在阻塞的时候会被信号中断。

#include <stdio.h>
#include <sys/select.h>
#include <stdlib.h>
#include <signal.h>
/* SIGUSR1的信号处理函数 */
void sigusr1_handler(int signo)
{
printf("catch SIGUSR1\n"); /* 接收SIGUSR1信号,打印接收信息 */
}
int main()
{
int rdy; /* 准备好的设备数 */
/* 注册信号处理函数,如果捕捉到信号则输出提示信息 */
if(signal(SIGUSR1, sigusr1_handler) == SIG_ERR){
perror("can’t set handler for SIGUSR1");
exit();
}
/* 不关心所有的设备准备状态,所以检查设备的最大文件描述符的值也不再有意义。
* 等待时间结构为NULL,表示将等待时间设置为无限等待
*/
rdy = select(, NULL, NULL, NULL, NULL);
/* 因为是无限等待,所以绝对不应该执行到这里,输出提示信息 */
printf("should never be here\n");
return ;
}

第二个程序使用pselect函数,程序中人为的制造了一个造成函数阻塞的条件。该程序在阻塞的时候不会被信号中断。

#include <stdio.h>
#include <sys/select.h>
#include <stdlib.h>
#include <signal.h>
/* SIGUSR1的信号处理函数 */
void sigusr1_handler(int signo)
{
printf("catch SIGUSR1\n"); /* 接收SIGUSR1信号,打印接收信息 */
}
int main()
{
int rdy; /* 准备好的设备数 */
sigset_t set; /* 信号集 */
/* 注册信号处理函数,如果捕捉到信号则输出提示信息 */
if(signal(SIGUSR1, sigusr1_handler) == SIG_ERR){
perror("can’t set handler for SIGUSR1");
exit();
}
sigfillset(&set); /* 设置信号集,屏蔽所有的信号,包括SIGKILL和SIGSTOP */
/* 不关心所有的设备准备状态,所以检查设备的最大文件描述符的值也不再有意义。
* 等待时间结构为NULL,表示将等待时间设置为无限等待
*/
rdy = pselect(, NULL, NULL, NULL, NULL, &set);
/* 因为是无限等待,所以绝对不应该执行到这里,输出提示信息 */
printf("should never be here\n");
return ;
}

在调试的时候用kill命令向进程发送SIGUSR1信号,观察输出结果,通过两个程序运行的对比可以清楚的理解pselect函数和select函数的区别。

上一篇:A1128. N Queens Puzzle


下一篇:kubernetes 测试 Mariadb gtid 主从复制.