0

我正在尝试按照问题Make children process wait until received parent's signal的答案开发一个简单的铁路模拟。

我的任务:我正好有 5 个进程代表火车。我需要通过 创建这 5 个进程(T1、T2、T3、T4 和 T5)fork(),并暂停每个进程,直到所有进程都创建完毕。之后,父进程将向子进程发送信号,每个子进程将使用execl execl(execl_path_name, CHILDETCONE, i, NULL);)。发出信号后,父母等待所有孩子完成他们的任务。

我很了解处理函数,但我不清楚以下几点:

  1. 我需要execl在处理程序函数中插入我的内容吗?

  2. 从上一个问题的答案中,我不明白最后一个循环的重要性:

    for (int i = 0; i < NUMBER_TRAINS; i++)
           {
               wait(NULL);
           }
    

这是我的代码:

#include <stdio.h>
#include <signal.h>
#include <sys/wait.h>
#include "accessory.h"


#define NUMBER_TRACKS 16
#define NUMBER_STATIONS 8
#define NUMBER_TRAINS 5

#define TRACKS_INITIALS "MA"
#define STATION_INITIALS "S"
#define SIZE 256
#define CHILDETCONE "childETCone"

void handler(int sig);

int main(int argc , char *argv[]) {
    pid_t pid;
    pid_t pid_array[NUMBER_TRAINS];

    char track_name[2];
    char track_number[2];
    int execl_return;
    char str[2];
    char * execl_path_name;

    memset(pid_array, 0, sizeof(pid_array));

    /* create the MAx file initialized to zero */
    for (int i = 1; i < (NUMBER_TRACKS + 1); i++) {
        memset(track_name, '\0', sizeof(track_name));
        memset(track_number, '\0', sizeof(track_number));
        strcpy(track_name, TRACKS_INITIALS);
        sprintf(track_number, "%d", i);
        strcat(track_name, track_number);
        create_track_file(track_name, "", SIZE);
    }

    execl_path_name = get_file_name(CHILDETCONE, "", SIZE);
    printf("path %p\n", execl_path_name);


    for(int i = 0; i < NUMBER_TRAINS; i++) {
        pid = fork();

        if (pid < 0) {
            perror("fork");
            exit(1);
        }
        if (pid == 0) { //child
            //sprintf(str, "%d", i+1);
            //execl_return = execl(execl_path_name, CHILDETCONE, i, NULL);
            signal(SIGUSR1, handler);
            pause();
            exit(0);
        }
        //parent
        pid_array[i] = pid;
    }

    for (int j = 0; j < NUMBER_TRAINS; j++) {
        kill(pid_array[j], SIGUSR1);
        sleep(1);
    }

    for (int i = 0; i < NUMBER_TRAINS; i++) {
        wait(NULL);
    }

    return 0;
}

void handler(int sig) {
    printf("printed from child [%d]\n", getpid());
    printf("signal [%d]\n", sig);
}
4

1 回答 1

1

我需要execl在处理程序函数中插入我的内容吗?

No. pause()只有在调用它的进程捕获到导致信号处理函数运行的信​​号后才会返回。因此,execl呼叫可以在pause呼叫之后进行。我认为在你的情况下会更清楚。

还要注意,POSIX 标准化了“异步信号安全”函数的列表,这些函数对于信号处理程序调用是安全的,并且对于处理程序调用其他函数是不安全的 execl在列表中,但printf其他流 I/O 函数不在. 信号处理程序不应调用printf. 您的特定信号处理程序根本不需要做任何事情。

此外,请考虑使用sigsuspend()代替pause(),因为前者可以让您更好地控制哪些信号会导致您的火车启动。

从上一个问题的答案中,我不明白最后一个循环的重要性:

for (int i = 0; i < NUMBER_TRAINS; i++)
       {
           wait(NULL);
       }

wait()函数指示调用进程阻塞,直到其子进程之一终止。循环的wait()调用次数与子程序的数量一样多,因此,在没有错误的情况下,主程序不会继续,直到其所有子程序都终止。

看起来你可能试图通过sleep()在循环中调用来实现类似的东西kill,但这种策略是完全错误的。首先,每次都等待kill意味着孩子们的execl电话至少会sleep间隔一段时间,这不是我理解你想要的。其次,您无法提前知道孩子们需要多长时间才能完成,因此在某些情况下,您允许的一秒钟可能还不够。第三,由于您似乎期望孩子们跑得很快,所以一秒钟可能比您大部分时间需要的多得多。

于 2018-08-06T14:07:40.077 回答