popen2()在c中如何工作?

far*_*ane 5 c fork pipe dup2

我尝试使用管道、叉子和 dup 在我的程序中执行 md5sume 命令。我发现总和代码运行成功,但我无法理解某些代码行。这是我的代码:

int infp, outfp;

char buf[128];

if (popen2("md5sum", &infp, &outfp) <= 0)

  {

    printf("Unable to exec sort\n");

    exit(1);

  }

write(infp, "hello\n", 2);

close(infp);

*buf = '\0';

read(outfp, buf, 128);

printf("buf = '%s'\n", buf);

return 0;

}
Run Code Online (Sandbox Code Playgroud)
int p_stdin[2], p_stdout[2];

pid_t pid;

if (pipe(p_stdin) != 0 || pipe(p_stdout) != 0)

  return -1;

pid = fork();

if (pid < 0)

  return pid;

if (pid == 0)

  {

    close(p_stdin[WRITE]);

    dup2(p_stdin[READ], READ);

    close(p_stdout[READ]);

    dup2(p_stdout[WRITE], WRITE);

    execl("/bin/sh", "sh", "-c", command, NULL);

    perror("execl");

    exit(1);

  }

else 

  {

if (infp == NULL)

    close(p_stdin[WRITE]);

else

    *infp = p_stdin[WRITE];

if (outfp == NULL)

    close(p_stdout[READ]);

else

    *outfp = p_stdout[READ];
  }

return pid;

}
Run Code Online (Sandbox Code Playgroud)

我不明白 popen 函数。这条线到底是做什么的?

*infp = p_stdin[WRITE];
Run Code Online (Sandbox Code Playgroud)

管道之间如何通信?

Kar*_*uru 2

我不明白 popen 函数。

管道之间如何通信?

pipeline() :管道是单向的,是内核中的字节流缓冲区。由于它是字节流类型,写入者可以写入任意数量的字节,读取者可以读出任意数量的字节。但是,请注意,顺序读取是可能的,但查找(如 lseek)是不可能的。由于管道是单向的,写入管道的数据将在内核中缓冲,直到从管道的读取端读取。另外,如果管道已满,写入就会阻塞。

假设 fd 是一个由 2 个文件描述符组成的整数数组 (int fd[2]),那么 pipeline(fd) 系统调用应创建一个管道并返回一对文件描述符,使得 fd[1] (stdout 为 1) fd[0](stdin 为 0)应为管道的写端,fd[0](stdin 为 0)应为管道的读端。与命名管道(如 FIFO - 文件系统中具有名称的管道)不同,匿名管道只能在相关进程(如父子进程)之间使用。因此,应进行fork以在子进程中复制这2个父文件描述符,从而父进程与子进程共享管道,以便子进程应在写端写入,而父进程应从管道的读端读取,或者父进程应写入写端end 和 child 应从管道的 read-end 读取。应注意确保父级或子级根据场景关闭未使用的读取(fd[0])文件描述符/未使用的写入(fd[1])文件描述符。

popen() : popen 使您能够将另一个程序作为新进程调用,从而向其传输数据或从中接收数据。如果是 popen,请注意数据流的方向基于第二个参数。我们不需要手动创建子进程,因为 popen 会自动 forks 创建子进程,启动 shell 并执行通过 popen 传递的命令参数。它还根据类型参数自动在父级和子级之间建立适当的读取或写入流。

因此,popen() 简化了事情,因为它避免了手动调用/调用 pipeline、fork、exec 的需要,并简化了根据参数类型自动在父/子之间建立适当的流。然而,popen 的另一面是,应该注意的是,每次调用 popen() 都会导致创建额外的进程 - 也就是说,除了正在调用的程序之外,每次都会调用 shell -反过来会导致高资源消耗。