分叉一个孩子,从控制台阅读

时间:2013-01-29 19:28:03

标签: c linux fork alarm

ac程序,它会分叉一个孩子并从键盘获取int数字达10秒,10秒后停止读取并检查是否读取了任何数字,如果没有读取任何数字,它会终止儿子,否则它会打印出来读取屏幕上的数字并终止,它不会显示结果。

#include<stdio.h>
#include<stdlib.h>
#include<sys/types.h>
#include<unistd.h>
#include<signal.h>
#include<string.h>
#include<sys/wait.h>
#include<sys/errno.h>



int status,j,i=0;
pid_t son1, son2;
int string[20];
char s[5];


void handler(int signum)
{
if(i==0)
kill(son1,SIGKILL);
}

void signal_handler_1(int sig_num)
{
  printf("\nI sent the signal SIGUSR1 to my child\n");

}


main(){

    son1=fork();
     if(son1==0)
     {
        signal(SIGUSR1 , signal_handler_1);
        printf("I'm son1, my id is: %d\n", getpid());

      pause();
        printf("I'm son1 exitting: %d\n", getpid());

      exit(1);

    }
    if(son1>0){

    printf("I'm father, my id is: %d\n", getpid());
    signal(SIGALRM,handler);

         alarm(10);//how can i terminate the reading process after alarm??

               while(fgets(s,5,stdin)!=NULL)
                {
                  sscanf(s,"%d",&string[i]);
                  i++;
                }


              for(j=0;j<i;j++)
                printf("%d\n",string[j]);

              kill(son1,SIGUSR1);


              wait(&status);


    }
    return 0;
}

1 个答案:

答案 0 :(得分:0)

在为SIGALRM运行信号handeler之后,父级返回到while循环。

另外,像linkdd所说,请更详细地描述问题。

在你的代码中,父母将SIGKILL发送给孩子并且它变得不存在,即父母被卡住后的“僵尸”。父母从未注意到10秒已经过去并且永远读取输入。

不是最漂亮的方式,但这会使它发挥作用。

定义全局

int alarm_received = 0;

编辑父信号handeler以在闹钟响起时设置标志。

void handler(int signum)
{
    if(i==0){
        kill(son1,SIGUSR1);
    }
    alarm_received = 1;
}

儿童信号handeler可能是这样的。在信号处理程序中调用printf是不好的。

void signal_handler_1(int sig_num)
{
    char msg[] = "Child received SIGUSR1\n";
    write(STDOUT_FILENO, msg, sizeof(msg));
}

更改while循环以检查信号是否已发送。所以父母不会卡住。

while(alarm_received == 0)
{
    fgets(s,5,stdin);
    sscanf(s,"%d",&string[i]);
    i++;
}

它仍然非常混乱。 fgets()很可能在接收到信号后需要(取决于操作系统)空行,因为操作系统重新启动了read()(内部fgets)。解析输入也非常脆弱。