4

我正在编写一个程序,一旦按下按钮,我必须执行一个服务器进程(只有当我决定杀死他时才会停止)。
为了执行这个过程,我决定使用 fork/execv 机制:

void Command::RunServer() {

    pid = fork();

    if (pid==0) {
        chdir("./bin");
        char str[10];
        sprintf(str,"%d",port);
        char *argv[] = {"./Server", str};
        execv("./Server",argv);
    }
    else {
        config->pid = pid;
        return;
    }
}

在“按钮按下”的方法中,我这样做:

command->RunServer();

几天前它似乎工作得很好......现在我收到错误:

main: xcb_io.c:221: poll_for_event: Assertion `(((long) (event_sequence) - (long) (dpy->request)) <= 0)' failed.

我应该尝试切换到 pthread 吗?我做了什么坏事吗?

谢谢,

4

2 回答 2

5

当您执行fork()此操作时,您的进程的所有文件描述符都将复制到新的文件描述符中。并且当你这样做时,exec*()所有文件描述符也会被保留,除非它们被标记为 flag FD_CLOEXEC

我的猜测是某些库(可能是 Xlib)使用的某些 fd 被新进程继承,并且重复导致程序混乱。

在这些情况下,如果您想保持标准 I/O 处于打开状态,则 BSD 函数closefrom()( )很有用。closefrom(3)不幸的是,在 linux 中没有这样的功能,所以你必须做一个封闭的循环或类似的事情:

int open_max = sysconf (_SC_OPEN_MAX);
for (int i = 3; i < open_max; i++)
    close(i);

您可以在此处阅读有关此问题的更多信息。

于 2012-07-19T08:37:07.903 回答
3

在对 , 的调用中execvargv必须以空指针终止。前一行应该是:

char* argv[] = { "./Server", str, NULL };
于 2012-07-19T09:21:19.803 回答