2

我在运行以下代码时遇到问题,该代码调用 getutent() 来计算当前登录到系统的用户总数。计时器将每 1 秒调用一次,并将名为“isSigAlrmOccured”的布尔值设置为 true 并退出。主函数通过检查此布尔值来检查是否传递了计时器信号并监控登录用户的数量。不幸的是,定时器信号仅正确传递到主程序两次,之后我没有收到任何进一步的信号。在前两个信号之后,暂停函数调用不会被中断。

#include <stdio.h>
#include <signal.h>
#include <unistd.h>
#include <utmp.h>
#include <errno.h>

static int isSigAlrmOccured;

void alarm_handler (int signo)
{
  static int i=1;
  printf("\n Signal Occurred %d times\n",i++);
  isSigAlrmOccured = 1;

}

int main (int argc, char *argv[]) {
   struct itimerval delay;
   struct utmp *utmpstruct;
   int numuser;


   int ret;
   signal (SIGALRM, alarm_handler);
   delay.it_value.tv_sec = 1;
   delay.it_value.tv_usec = 0;
   delay.it_interval.tv_sec = 1;
   delay.it_interval.tv_usec = 0;
   ret = setitimer (ITIMER_REAL, &delay, NULL);
   if (ret) {
     perror ("setitimer");
     return 0;
   }
   for (;;) {
       pause ( );
       /* count the number of users */
       if ( (errno == EINTR) && (isSigAlrmOccured) ) {
           isSigAlrmOccured = 0;
           setutent();
           while ((utmpstruct = getutent())) {
           if ((utmpstruct->ut_type == USER_PROCESS) &&
               (utmpstruct->ut_name[0] != '\0'))
               numuser++;
           }
           endutent();
       }
   }
   return 0;
}

输出:

信号发生1次

信号发生 2 次

4

2 回答 2

7

*utent() 的实现正在使用 alarm() 并正在重置您的警报。
你将不得不做其他事情。

strace ttest(为简洁起见删除了一些行)
[...]
pause()
--- SIGALRM(闹钟)@ 0 (0) ---
write(1, "Signal Occurred 1 times\n", 25 Signal Occurred 1次) = 25
打开("/var/run/utmp", O_RDONLY|O_CLOEXEC) = 3
警报 (0) = 5
rt_sigaction(SIGALRM, {0x7f52580a91c0, [], SA_RESTORER, 0x7f5257fd46e0}, {0x40075c, [ALRM], SA_RESTORER |SA_RESTART, 0x7f5257fd46e0}, 8) = 0
警报 (1) = 0

仅在睡眠期间设置警报的示例代码。

#include <stdio.h>
#include <signal.h>
#include <unistd.h>
#include <utmp.h>
#include <errno.h>

static int isSigAlrmOccured;

void alarm_handler (int signo)
{
  static int i=1;
  printf("\n Signal Occurred %d times\n",i++);
  isSigAlrmOccured = 1;

}

int main (int argc, char *argv[]) {
   struct itimerval delay;
   struct utmp *utmpstruct;
   int numuser;
   int ret;

   for (;;) {
   signal (SIGALRM, alarm_handler);
       alarm(1);       /* wake me later */
       pause ( );
       /* count the number of users */
       if ( (errno == EINTR) && (isSigAlrmOccured) ) {
       signal (SIGALRM, SIG_DFL);
           isSigAlrmOccured = 0;
           numuser = 0;
           setutent();
           while ((utmpstruct = getutent())) {
           if ((utmpstruct->ut_type == USER_PROCESS) &&
               (utmpstruct->ut_name[0] != '\0'))
               numuser++;
           }
           endutent();
           printf("found %d users\n", numuser);
       }
   }
   return 0;
}
于 2009-08-26T12:25:02.250 回答
1

此页面列出了从信号处理程序调用“安全”的一组函数。如果您调用其他函数,则行为未定义。我注意到这setutent()似乎并不是最重要的,对于初学者来说......

于 2009-08-26T11:51:58.890 回答