我正在尝试设置一些信号处理程序。具体来说,我想安装一个SIGQUIT处理程序,它恢复默认处理,但在n秒后重新安装SIGQUIT处理程序。
当我最初在main()中安装我的信号处理程序时没有问题,但是当我添加代码以尝试在我的SIGALRM处理程序中重新安装我的SIGQUIT处理程序时,我在编译时不断收到以下错误消息:
ish.c: In function ‘SIGALRM_Handler’:
ish.c:42: error: ‘SIGQUIT_Handler’ undeclared (first use in this function)
ish.c:42: error: (Each undeclared identifier is reported only once
ish.c:42: error: for each function it appears in.)
以下是代码的相关摘要。我想要做的就是在触发警报时重新安装SIGQUIT处理程序,但我不确定如何继续?
/* SIGALRM handler */
static void SIGALRM_Handler(int iSig)
{
void (*pfRet)(int);
/* reinstall SIGQUIT handler */
pfRet = signal(SIGQUIT, SIGQUIT_Handler);
if (pfRet == SIG_ERR) { perror("SIGQUIT_Handler"); exit(EXIT_FAILURE); }
}
/*--------------------------------------------------------------------*/
/* SIGQUIT handler */
static void SIGQUIT_Handler(int iSig)
{
void (*pfRet)(int);
/* Print message followed by shell prompt */
printf("Type Ctrl-\\ again within 5 seconds to exit\n");
printf("------------------------------------\n");
printf("%c ", '%');
fflush(NULL);
/* install default SIGQUIT handling */
pfRet = signal(SIGQUIT, SIG_DFL);
if (pfRet == SIG_ERR) { perror("SIGQUIT_Handler"); exit(EXIT_FAILURE); }
/* set alarm for 5 seconds later */
alarm(5);
}
...
int main(int argc, char **argv)
{
FILE *psFile;
void (*pfRet)(int);
sigset_t sSet;
int iRet;
assert(argv != NULL);
/* make sure needed signals are not blocked */
sigemptyset(&sSet);
sigaddset(&sSet, SIGINT);
sigaddset(&sSet, SIGQUIT);
sigaddset(&sSet, SIGALRM);
iRet = sigprocmask(SIG_UNBLOCK, &sSet, NULL);
if (iRet == -1) { perror(argv[0]); exit(EXIT_FAILURE); }
/* install SIGINT handler */
pfRet = signal(SIGINT, SIGINT_Handler);
if (pfRet == SIG_ERR) { perror(argv[0]); exit(EXIT_FAILURE); }
/* install SIGALRM handler */
pfRet = signal(SIGALRM, SIGALRM_Handler);
if (pfRet == SIG_ERR) { perror(argv[0]); exit(EXIT_FAILURE); }
/* install SIGQUIT handler */
pfRet = signal(SIGQUIT, SIGQUIT_Handler);
if (pfRet == SIG_ERR) { perror(argv[0]); exit(EXIT_FAILURE); }
/* check for configuration file */
if (argc > 2)
{
fprintf(stderr, "Usage: ish [configfile]\n");
exit(EXIT_FAILURE);
}
else if (argc == 2)
{
psFile = fopen(argv[1], "r");
if (psFile != NULL)
{
handleProcess(psFile);
}
else
{
fprintf(stderr, "ish: ");
perror(argv[1]);
}
}
for(;;)
{
handleProcess(stdin);
}
return 0;
}
答案 0 :(得分:2)
您需要在SIGQUIT_Handler()
中将SIGALRM_Handler()
的定义放在引用之前,或者在它之前添加声明:
static void SIGQUIT_Handler(int iSig);
要么可以工作,但有一个是必要的。为什么?当你写:
signal(SIGQUIT, SIGQUIT_Handler);
第二个参数必须是指向函数的指针,但编译器必须知道该名称表示指向函数的指针,因为它在使用之前已声明。从根本上说,名称SIGQUIT_Handler
从未被声明或定义,因此编译器不知道它是什么。