⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 14.htm

📁 UNIX环境下C编程的详细详细介绍
💻 HTM
📖 第 1 页 / 共 5 页
字号:
</p>

<p>的读操作,所以这不会产生任何影响。 </p>

<p>#include &quot;ourhdr.h&quot; </p>

<p>static int pfd1[2], pfd2[2]; </p>

<p>void </p>

<p>TELL_WAIT() </p>

<p>{ </p>

<p>if (pipe(pfd1) &lt; 0 || pipe(pfd2) &lt; 0) </p>

<p>err_sys(&quot;pipe error&quot;); </p>

<p>} </p>

<p>void </p>

<p>TELL_PARENT(pid_t pid) </p>

<p>{ </p>

<p>if (write(pfd2[1], &quot;c&quot;, 1) != 1) </p>

<p>err_sys(&quot;write error&quot;); </p>

<p>} </p>

<p>void </p>

<p>WAIT_PARENT(void) </p>

<p>{ </p>

<p>char c; </p>

<p>if (read(pfd1[0], &amp;c, 1) != 1) </p>

<p>err_sys(&quot;read error&quot;); </p>

<p>if (c != 'p') </p>

<p>err_quit(&quot;WAIT_PARENT: incorrect data&quot;); </p>

<p>} </p>

<p>void </p>

<p>TELL_CHILD(pid_t pid) </p>

<p>{ </p>

<p>if (write(pfd1[1], &quot;p&quot;, 1) != 1) </p>

<p>err_sys(&quot;write error&quot;); </p>

<p>} </p>

<p>void </p>

<p>WAIT_CHILD(void) </p>

<p>{ </p>

<p>char c; </p>

<p>if (read(pfd2[0], &amp;c, 1) != 1) </p>

<p>err_sys(&quot;read error&quot;); </p>

<p>if (c != 'c') </p>

<p>err_quit(&quot;WAIT_CHILD: incorrect data&quot;); </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 &lt;stdio.h&gt; </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是&quot;r&quot;,则文件指针连到cmdstring的标准输出(图14.6) 
</p>

<p>图14.6 fp=popen(command , &quot;r&quot;)的结果 </p>

<p>如果type 是 &quot;w&quot;,则文件指针连接到cmdstring 的标准输入(图14.7)。 
</p>

<p>图14.7 fp=popen(command,&quot;w&quot;)的结果 </p>

<p>有一种方法可以帮助我们记住popen最后一个参数及其作用,这种方法就是与fope 
</p>

<p>n进行类比。如果type是&quot;r&quot;,则返回的文件指针是可读的,如果type是&quot;w&quot;,则是可 
</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(&quot;ls *.c&quot; , &quot;r&quot;); </p>

<p>或者 </p>

<p>fp=popen(&quot;cmd 2&gt;&amp;1&quot; , &quot;r&quot;); </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是&quot;r&quot;,则文件指针连到cmdstring的标准输出(图14.6) 
</p>

<p>图14.6 fp=popen(command , &quot;r&quot;)的结果 </p>

<p>如果type 是 &quot;w&quot;,则文件指针连接到cmdstring 的标准输入(图14.7)。 
</p>

<p>图14.7 fp=popen(command,&quot;w&quot;)的结果 </p>

<p>有一种方法可以帮助我们记住popen最后一个参数及其作用,这种方法就是与fope 
</p>

<p>n进行类比。如果type是&quot;r&quot;,则返回的文件指针是可读的,如果type是&quot;w&quot;,则是可 
</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(&quot;ls *.c&quot; , &quot;r&quot;); </p>

<p>或者 </p>

<p>fp=popen(&quot;cmd 2&gt;&amp;1&quot; , &quot;r&quot;); </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 &lt;sys/wait.h&gt; </p>

<p>#include &quot;ourhdr.h&quot; </p>

<p>#define PAGER &quot;${PAGER:-more}&quot; /* 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(&quot;usage: a.out &lt;pathname&gt;&quot;); </p>

<p>if ( (fpin = fopen(argv[1], &quot;r&quot;)) == NULL) </p>

<p>err_sys(&quot;can't open %s&quot;, argv[1]); </p>

<p>if ( (fpout = popen(PAGER, &quot;w&quot;)) == NULL) </p>

<p>err_sys(&quot;popen error&quot;); </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(&quot;fputs error to pipe&quot;); </p>

<p>} </p>

<p>if (ferror(fpin)) </p>

<p>err_sys(&quot;fgets error&quot;); </p>

<p>if (pclose(fpout) == -1) </p>

<p>err_sys(&quot;pclose error&quot;); </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 &lt;sys/wait.h&gt; </p>

<p>#include &lt;errno.h&gt; </p>

<p>#include &lt;fcntl.h&gt; </p>

<p>#include &quot;ourhdr.h&quot; </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 &quot;/bin/sh&quot; </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 &quot;r&quot; or &quot;w&quot; */ </p>

<p>if ((type[0] != 'r' &amp;&amp; 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) &lt; 0) </p>

<p>return(NULL); /* errno set by pipe() */ </p>

<p>if ( (pid = fork()) &lt; 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, &amp;stat, 0) &lt; 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 + -