?? 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 + -