c++ - fork() 输出链

标签 c++ unix fork netbeans-7 g++-4.7

我在 Unix 环境中完全是个新手,我遇到了一些来自 Robbins 的 Unix Systems Programming 一书中的简单示例的问题。

这是简单的进程链,每个进程打印一些信息到日志文件和标准错误

#define BUFSIZE 1024
#define CREATE_FLAGS (O_WRONLY | O_CREAT | O_APPEND)
#define CREATE_PERMS (S_IRUSR | S_IWUSR| S_IRGRP | S_IROTH)

int main  (int argc, char *argv[]) {
   char buf[BUFSIZE];
   pid_t childpid = 0;
   int i, n;
   if (argc != 3){       /* check for valid number of command-line arguments */
      fprintf (stderr, "Usage: %s processes filename\n", argv[0]);
      return 1;
   }
                                        /* open the log file before the fork */

   n = atoi(argv[1]);                              /* create a process chain */
   for (i = 1; i < n; i++)
       if (childpid = fork())
          break;
   if (childpid == -1) {
      fprintf(stderr, "Failed to fork");
      return 1;
   }

   auto fd = open(argv[2], CREATE_FLAGS, CREATE_PERMS);
   if (fd < 0) {
      fprintf(stderr,"Failed to open file");
      return 1;
   }

   sprintf(buf, "i:%d process:%ld parent:%ld child:%ld\n", 
           i, (long)getpid(), (long)getppid(), (long)childpid);
   fprintf(stderr, buf);
   write(fd, buf, strlen(buf));
   return 0;
}

它是在 Netbeans 7.1 上用 g++ 4.7 编译的,运行命令是 "${OUTPUT_PATH}"10/home/maxim/teSTLog.log

所以问题是:

  1. 当我运行或调试项目时,它在控制台和文件中仅打印出 2 或 3 行信息。但是,如果我通过 childpid = fork() 以“Step Over”遍历,它会打印有关所有 10 个进程的信息。这是一些编译器优化还是我的错?

  2. 即使它打印所有行,输出看起来也是这样

    i:2 process:6571 parent:6566 child:6572
    i:3 process:6572 parent:1 child:6573
    i:4 process:6573 parent:6572 child:6574
    ...
    i:9 process:6578 parent:1 child:6579
    i:10 process:6579 parent:6578 child:0
    

    有些进程的父pid值为1,好像是错误的

最佳答案

  1. 如果每个进程都打开相同的输出文件,则会出现竞争条件,导致进程相互覆盖。这就是为什么只有全速奔跑时才会发生这种情况。

  2. 当父进程结束时,任何仍然存活的子进程要么被杀死,要么获得一个新的父进程,具体取决于 Linux 中的设置。在你的情况下,他们似乎有了新的 parent 。那个新父进程是进程 1。

关于c++ - fork() 输出链,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20241046/

相关文章:

c++ - 在 Visual Studio 2012 中从一个项目引用另一个项目中的 DLL

unix - docker无法从DockerFile启动服务

c - 关于 fork() 的问题

git - 将 Github 分支添加到现有存储库

c++ - RapidJSON 如何在使用 FindMember 时接收成员对象

c++ - 连接c++文件和gnuplot并在执行后自动绘图

c++ - 如何清除 cin 中的无关字符?

linux - 将旧文件更改为时间戳的脚本

unix - 以不同语言打印大纪元时间

c++ - 如何知道自己进程打开的各种文件是什么