在while(1)
循环中使用pause()
函数,实际上是一个常见的实现方式,通常被称为空闲循环或轮询循环。在这种循环中,程序会无限循环执行两个步骤:等待信号并处理信号。
具体的执行过程如下:
while(1)
是一个无限循环,条件表达式永远为真,因此循环会一直执行。- 在每次循环迭代中,程序会调用
pause()
函数。 pause()
函数会将进程挂起,直到接收到一个信号。- 当接收到信号时,
pause()
函数返回,并且程序继续执行。通常,在接收到信号后,会调用相应的信号处理函数来处理信号。 - 执行完信号处理函数后,程序再次回到循环开始的地方,继续执行下一次循环。
这种循环模式可以用于处理异步事件或等待特定条件的发生。通过调用pause()
函数来挂起程序,可以有效地防止循环的空转,减少不必要的处理器资源占用。
需要注意的是,在这种循环中,如果没有信号被发送或没有信号处理函数来处理信号,pause()
函数可能会一直挂起进程,导致程序无法继续执行。因此,在使用while(1)
和pause()
循环时,确保有适当的信号处理机制来保证程序的正常执行。
比如下面的程序:
用信号的知识实现司机和售票员问题。
1)售票员捕捉SIGINT(代表开车)信号,向司机发送SIGUSR1信号,司机打印(let's gogogo)
2)售票员捕捉SIGQUIT(代表停车)信号,向司机发送SIGUSR2信号,司机打印(stop the bus)
3)司机捕捉SIGTSTP(代表到达终点站)信号,向售票员发送SIGUSR1信号,售票员打印(please get off the bus)
4)司机等待售票员下车,之后司机再下车。
父子进程:
子:售票员:
忽略:SIGTSTP
捕捉:SIGINT--》kill---》SIGUSR1
SIGQUIT---》kill----》SIGUSR2
SIGUSR1---》printf---》exit
父:司机:
忽略:SIGINT SIGQUIT
捕捉:SIGUSR1----》printf
SIGUSR2----》printf
SIGTSTP---》kill---》SIGUSR1---》wait---》exit
#include <stdio.h>
#include <signal.h>
#include <sys/wait.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>
__pid_t pid;
void kill1(int sig)
{
if (sig == SIGINT)
if (kill(getppid(), SIGUSR1) == -1)
{
perror("kill");
}
}
void kill2(int sig)
{
if (sig == SIGQUIT)
if (kill(getppid(), SIGUSR2) == -1)
{
perror("kill");
}
}
void kill3(int sig)
{
if (sig == SIGTSTP)
{
if (kill(pid, SIGUSR1) == -1)
{
perror("kill");
}
else
{
wait(NULL);
exit(0);
}
}
}
void printf1(int sig)
{
if (sig == SIGUSR1)
printf("let's gogogo\n");
}
void printf2(int sig)
{
if (sig == SIGUSR2)
printf("stop the bus\n");
}
void printf3(int sig)
{
if (sig == SIGUSR1)
{
printf("please get off the bus\n");
exit(0);
}
}
int main(int argc, char const *argv[])
{
signal(SIGINT, SIG_IGN);
signal(SIGUSR1, printf1);
signal(SIGQUIT, SIG_IGN);
signal(SIGUSR2, printf2);
signal(SIGTSTP, kill3);
pid = fork();
if (pid < 0)
{
perror("fork");
return -1;
}
else if (pid == 0)
{
signal(SIGINT, kill1);
// signal(SIGUSR1, SIG_IGN);
signal(SIGQUIT, kill2);
// signal(SIGUSR2, SIG_IGN);
signal(SIGTSTP, SIG_IGN);
signal(SIGUSR1, printf3);
while (1)
pause();
}
// signal(SIGUSR1, SIG_IGN);
while (1)
pause();
return 0;
}