📄 14.htm
字号:
</p>
<p>的读操作,所以这不会产生任何影响。 </p>
<p>#include "ourhdr.h" </p>
<p>static int pfd1[2], pfd2[2]; </p>
<p>void </p>
<p>TELL_WAIT() </p>
<p>{ </p>
<p>if (pipe(pfd1) < 0 || pipe(pfd2) < 0) </p>
<p>err_sys("pipe error"); </p>
<p>} </p>
<p>void </p>
<p>TELL_PARENT(pid_t pid) </p>
<p>{ </p>
<p>if (write(pfd2[1], "c", 1) != 1) </p>
<p>err_sys("write error"); </p>
<p>} </p>
<p>void </p>
<p>WAIT_PARENT(void) </p>
<p>{ </p>
<p>char c; </p>
<p>if (read(pfd1[0], &c, 1) != 1) </p>
<p>err_sys("read error"); </p>
<p>if (c != 'p') </p>
<p>err_quit("WAIT_PARENT: incorrect data"); </p>
<p>} </p>
<p>void </p>
<p>TELL_CHILD(pid_t pid) </p>
<p>{ </p>
<p>if (write(pfd1[1], "p", 1) != 1) </p>
<p>err_sys("write error"); </p>
<p>} </p>
<p>void </p>
<p>WAIT_CHILD(void) </p>
<p>{ </p>
<p>char c; </p>
<p>if (read(pfd2[0], &c, 1) != 1) </p>
<p>err_sys("read error"); </p>
<p>if (c != 'c') </p>
<p>err_quit("WAIT_CHILD: incorrect data"); </p>
<p>} </p>
<p>程序14.3 使父、子进程同步的例程 </p>
<p>14.3 popen和pclose函数 </p>
<p>因为常见的操作是创建一个连到另一个进程的管道,然后读其输出或向其发送输入
</p>
<p>,所以标准I/O库为实现这些操作提供了两个函数popen和pclose。这两个函数实现
</p>
<p>的操作是;创建一个管道,fork一个子进程,关闭管道的不使用端,exec一个she
</p>
<p>ll以执行命令,等待命令终止。 </p>
<p>#include <stdio.h> </p>
<p>FILE *popen(const char *cmdstring, const char *type); </p>
<p>返回:若成功为文件指针,出错为NULL </p>
<p>int pclose(FILE *fp); </p>
<p>返回:cmdstring? </p>
<p>终止状态,出错为-1 </p>
<p>函数popen 先做fork,然后exec以执行cmdstring,并且返回一个标准I/O文件指针
</p>
<p>。如果type是"r",则文件指针连到cmdstring的标准输出(图14.6)
</p>
<p>图14.6 fp=popen(command , "r")的结果 </p>
<p>如果type 是 "w",则文件指针连接到cmdstring 的标准输入(图14.7)。
</p>
<p>图14.7 fp=popen(command,"w")的结果 </p>
<p>有一种方法可以帮助我们记住popen最后一个参数及其作用,这种方法就是与fope
</p>
<p>n进行类比。如果type是"r",则返回的文件指针是可读的,如果type是"w",则是可
</p>
<p>写的。 </p>
<p>pclose函数关闭标准I/O流,等待命令执行结束,然后返回shell的终止状态。(我
</p>
<p>们曾在8.6节对终止状态进行过说明,system函数(8.12节)也返回终止状态。)
</p>
<p>如果shell不能被执行,则pclose返回的终止状态与shell执行exit(127)一样。
</p>
<p>cmdstring 由Bourne shell以下列方式执行; </p>
<p>sh -c cmdstring </p>
<p>这表示shell将扩展cmdstring中的任何特殊字符。例如,我们可以使用;
</p>
<p>fp=popen("ls *.c" , "r"); </p>
<p>或者 </p>
<p>fp=popen("cmd 2>&1" , "r"); </p>
<p>返回:若成功为文件指针,出错为NULL </p>
<p>int pclose(FILE *fp); </p>
<p>返回:cmdstring? </p>
<p>终止状态,出错为-1 </p>
<p>函数popen 先做fork,然后exec以执行cmdstring,并且返回一个标准I/O文件指针
</p>
<p>。如果type是"r",则文件指针连到cmdstring的标准输出(图14.6)
</p>
<p>图14.6 fp=popen(command , "r")的结果 </p>
<p>如果type 是 "w",则文件指针连接到cmdstring 的标准输入(图14.7)。
</p>
<p>图14.7 fp=popen(command,"w")的结果 </p>
<p>有一种方法可以帮助我们记住popen最后一个参数及其作用,这种方法就是与fope
</p>
<p>n进行类比。如果type是"r",则返回的文件指针是可读的,如果type是"w",则是可
</p>
<p>写的。 </p>
<p>pclose函数关闭标准I/O流,等待命令执行结束,然后返回shell的终止状态。(我
</p>
<p>们曾在8.6节对终止状态进行过说明,system函数(8.12节)也返回终止状态。)
</p>
<p>如果shell不能被执行,则pclose返回的终止状态与shell执行exit(127)一样。
</p>
<p>cmdstring 由Bourne shell以下列方式执行; </p>
<p>sh -c cmdstring </p>
<p>这表示shell将扩展cmdstring中的任何特殊字符。例如,我们可以使用;
</p>
<p>fp=popen("ls *.c" , "r"); </p>
<p>或者 </p>
<p>fp=popen("cmd 2>&1" , "r"); </p>
<p>POSIX.1没有说明popen、pclose,因为它们与shell有交互作用,而shell 是由POS
</p>
<p>IX.2说明的。我们对这两个函数的说明与POSIX.2的 Draft11.2(11.2草案)相一致
</p>
<p>。该POSIX.2草案对这两个函数的说明与以前的实现有些区别。 </p>
<p>实例#include <sys/wait.h> </p>
<p>#include "ourhdr.h" </p>
<p>#define PAGER "${PAGER:-more}" /* environment variable, or default */ </p>
<p>int </p>
<p>main(int argc, char *argv[]) </p>
<p>{ </p>
<p>char line[MAXLINE]; </p>
<p>FILE *fpin, *fpout; </p>
<p>if (argc != 2) </p>
<p>err_quit("usage: a.out <pathname>"); </p>
<p>if ( (fpin = fopen(argv[1], "r")) == NULL) </p>
<p>err_sys("can't open %s", argv[1]); </p>
<p>if ( (fpout = popen(PAGER, "w")) == NULL) </p>
<p>err_sys("popen error"); </p>
<p>/* copy argv[1] to pager */ </p>
<p>while (fgets(line, MAXLINE, fpin) != NULL) { </p>
<p>if (fputs(line, fpout) == EOF) </p>
<p>err_sys("fputs error to pipe"); </p>
<p>} </p>
<p>if (ferror(fpin)) </p>
<p>err_sys("fgets error"); </p>
<p>if (pclose(fpout) == -1) </p>
<p>err_sys("pclose error"); </p>
<p>exit(0); </p>
<p>} </p>
<p>程序14.4 用popen向分页程序传送文件 </p>
<p>让我们用popen重写程序14.2,其结果是程序14.4。使用popen减少了需要编写的代
</p>
<p>码量。 </p>
<p>shell命令${PAGER:-more}的意思是;如果shell变量PAGER已经定义,且其值非空
</p>
<p>(nonnull),则使用其值,否则使用字符串more。 </p>
<p>实例-popen函数 </p>
<p>程序14.5是我们编写的popen和pclose版本。虽然popen的核心部分与本章中以前用
</p>
<p>过的代码类似,但是增加了很多需要考虑的细节。首先每次调用popen时,应当记
</p>
<p>住所创建的子进程的进程ID,以及其文件描述符或FILE指针。我们选择在数组chi
</p>
<p>ldpid中保存子进程ID,并用文件描述符作为其下标。于是,当以FILE指针作为参
</p>
<p>数调用pclose时,我们调用标准I/O函数fileno以得到文件描述符,然后取得子进
</p>
<p>程ID,并用于调用waitpid。因为一个进程可能调用popen多次,所以我们在动态分
</p>
<p>配childpid数组时(第一次调用popen时),其长度可以容纳与文件描述符数相同
</p>
<p>的进程数。 </p>
<p>调用pipe、fork以及为每个进程复制相应的文件描述符,这些操作与本章前面所述
</p>
<p>的类似。 </p>
<p>POSIX.2要求子进程关闭在以前调用popen时形成,当前仍旧打开的所有I/O流。为
</p>
<p>此,我们在子进程中从头逐个检查childpid数组的各元素,关闭仍旧打开的任一描
</p>
<p>述符。 </p>
<p>若pclose的调用者已经为信号SIGCHLD设置了一个信号处理程序,则waitpid将返回
</p>
<p>一个出错号EINTR。因为允许调用者捕捉此信号(或者任何其它可能中断waitpid调
</p>
<p>用的信号),所以当waitpid被一个捕捉到的信号中断时,我们只是再次调用wait
</p>
<p>pid。 </p>
<p>如果一个信号中断了wait, pclose的早期版本返回EINTR出错号。 </p>
<p>pclose的早期版本在wait期间,阻塞或忽略信号SIGINT、SIGQUIT以及SI </p>
<p>GHUP。 </p>
<p>POSIX.2则不允许这一点。 </p>
<p>#include <sys/wait.h> </p>
<p>#include <errno.h> </p>
<p>#include <fcntl.h> </p>
<p>#include "ourhdr.h" </p>
<p>static pid_t *childpid = NULL; </p>
<p>/* ptr to array allocated at run </p>
<p>time */ </p>
<p>static int maxfd; /* from our open_max(), {Prog openmax} */ </p>
<p>#define SHELL "/bin/sh" </p>
<p>FILE * </p>
<p>popen(const char *cmdstring, const char *type) </p>
<p>{ </p>
<p>int i, pfd[2]; </p>
<p>pid_t pid; </p>
<p>FILE *fp; </p>
<p>/* only allow "r" or "w" */ </p>
<p>if ((type[0] != 'r' && type[0] != 'w') || type[1] != 0) { </p>
<p>errno = EINVAL; /* required by POSIX.2 */ </p>
<p>return(NULL); </p>
<p>} </p>
<p>if (childpid == NULL) { /* first time through */ </p>
<p>/* allocate zeroed out array for child pids */ </p>
<p>maxfd = open_max(); </p>
<p>if ( (childpid = calloc(maxfd, sizeof(pid_t))) == NULL) </p>
<p>return(NULL); </p>
<p>} </p>
<p>if (pipe(pfd) < 0) </p>
<p>return(NULL); /* errno set by pipe() */ </p>
<p>if ( (pid = fork()) < 0) </p>
<p>return(NULL); /* errno set by fork() */ </p>
<p>else if (pid == 0) { </p>
<p>* child */ </p>
<p>if (*type == 'r') { </p>
<p>if ( (fp = fdopen(pfd[0], type)) == NULL) </p>
<p>return(NULL); </p>
<p>} else { </p>
<p>close(pfd[0]); </p>
<p>if ( (fp = fdopen(pfd[1], type)) == NULL) </p>
<p>return(NULL); </p>
<p>} </p>
<p>childpid[fileno(fp)] = pid; /* remember child pid for this fd */ </p>
<p>return(fp); </p>
<p>} </p>
<p>int </p>
<p>pclose(FILE *fp) </p>
<p>{ </p>
<p>int fd, stat; </p>
<p>pid_t pid; </p>
<p>if (childpid == NULL) </p>
<p>return(-1); /* popen() has never been called */ </p>
<p>fd = fileno(fp); </p>
<p>if ( (pid = childpid[fd]) == 0) </p>
<p>return(-1); /* fp wasn't opened by popen() */ </p>
<p>childpid[fd] = 0; </p>
<p>if (fclose(fp) == EOF) </p>
<p>return(-1); </p>
<p>while (waitpid(pid, &stat, 0) < 0) </p>
<p>if (errno != EINTR) </p>
<p>return(-1); /* error other than EINTR from waitpid() </p>
<p>*/ </p>
<p>return(stat); /* return child's termination status */ </p>
<p>} </p>
<p>程序14.5 popen和pclose函数 </p>
<p>实例 </p>
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -