前段时间看《UNIX网络编程》的时候,想使用alarm中断像read、accept之类的阻塞调用。当时大体上是这样实现的:
signal(SIGALRM, handler);
struct sockaddr_in cliaddr;
socklen_t clilen = sizeof(cliaddr);
alarm(5);
int connfd = accept(listen_fd, (struct sockaddr *)&cliaddr, &clilen);
为了简单,使用的signal设置的信号处理函数,结果SIGALRM信号确实可以获取到,但是accept却没有被中断,当时我还以为是accept实现了自动重启的功能。之后想起来sigaction系统调用可以在sa_flags中设置SA_RESTART,也就是自动重启中断的系统调用的功能,才想到是不是signal默认设置了该flag。于是将代码改为了使用sigaction。
struct sigaction sa;
sa.sa_flags = 0;
sigemptyset(&sa.sa_mask);
sa.sa_handler = handler;
if (sigaction(SIGALRM, &sa, NULL) == -1)
{
perror("sigaction");
exit(EXIT_FAILURE);
}
alarm(5);
没有设置SA_RESTART,这样accept就可以被中断了。然后把sa.sa_flags设置为SA_RESTART,结果就跟之前signal一样无法中断accept了。
看来signal还是最好不要使用,因为难以确定其实际的表现。