我们应该在子进程中使用退出还是返回

Yve*_*ves 7 c c++ unix linux fork

说我曾经fork创建一个子进程。这是一个例子:

pid_t pid=fork();
if (pid==0) /* child */
{
    // do something
    exit(0); // _exit, exit or return????
}
else /* parrent */
{
    wait(nullptr);
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

我已经看到了许多有关的示例fork。其中一些用于_exit终止子进程,以避免刷新I / O缓冲区,另一些用于exit终止子进程。但是他们都不习惯return。正如我的理解,_exit并且exit不会自动调用析构函数,所以是它更好地调用return,而不是exit在子进程?还是因为我见过的所有示例都是C而不是C ++,所以它们不必担心析构函数?

bk2*_*204 3

您可以使用 或_exitexit但不应使用return. 当您分叉子项时,您将保留整个调用堆栈作为分叉子项的一部分。因此,如果您使用return,您最终会在程序中一路返回,可能会继续执行其他任务,这几乎肯定不是您想要的。

例如,如果您有类似以下代码片段的内容:

int get_value()
{
    pid_t pid;
    if (!(pid = fork())) {
        int x = 0;
        // do something with x.
        exit(x);
    }
    else {
        int status;
        wait(&status);
        return status;
    }
}

int main()
{
    int value = get_value();
    switch (get_value()) {
        case 0:
            // call f
            break;
        case 255 << 8:
            // call g
            break;
    }
}
Run Code Online (Sandbox Code Playgroud)

您最终可能会打电话fg与 一起做其他工作return,这绝对是不希望的。

如果您调用,则不会调用_exit注册的函数。atexit这是在线程环境中正确的做法。如果您不在线程环境中工作并且没有使用 注册任何处理程序atexit,那么它们在功能上应该是等效的。

如果您希望调用子进程中的析构函数,请将子进程代码放入其自己的函数中,并让其变量在超出范围时自动销毁。exit不会为您销毁对象,这很好,因为通常您不希望在子进程中销毁在父进程中创建的对象。