上一篇的管道通信可以看得出來,管道通信是單向的,只能一端寫入,一端讀取,有點類似通信中的單工通信,若要實現雙工通信,可以採用兩個單工通信。同樣的道理,要實現管道雙向通信,採用兩個管道即可。
上圖完成這樣一項任務:父進程負責輸入兩個數據x,y到管道,然後從管道讀取他們相加的返回結果,子進程從管道讀取x和y的值,執行add操作(這裏還是和上個程序一樣,採用exec函數來執行程序,所以必須先寫一個add.c文件,然後編譯鏈接程add文件)。這裏的add.c文件中,我們會選擇從標準輸入讀取數據,然後會把結果打印到標準輸出,所以在子進程中,需要做重定向操作。
源碼如下:
add.c
#include <unistd.h>
#include <string.h>
#include <stdio.h>
#include <stdlib.h>
int main(void)
{
int x,y;
if(read(STDIN_FILENO,&x,sizeof(int))<0)
{
perror("read error");
}
if(read(STDIN_FILENO,&y,sizeof(int))<0)
{
perror("read error");
}
int result=x+y;
if(write(STDOUT_FILENO,&result,sizeof(int))!=sizeof(int))
{
perror("write error");
}
return 0;
}
pipe.c
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
int main(void)
{
int fda[2],fdb[2];
if((pipe(fda)<0)||(pipe(fdb)<0)){
perror("pipe error");
exit(1);
}
pid_t pid;
pid=fork();
if(pid<0)
{
perror("fork error");
exit(1);
}
else if(pid==0)
{
close(fda[1]);
close(fdb[0]);
int x,y;
if(dup2(fda[0],STDIN_FILENO)!=STDIN_FILENO)
{
perror("dup2 error");
}
if(dup2(fdb[1],STDOUT_FILENO)!=STDOUT_FILENO)
{
perror("dup2 error");
}
if(execlp("./add","./add",NULL)<0)
{
perror("execvp error");
exit(1);
}
close(fda[0]);
close(fdb[1]);
}
else
{
int x,y;
printf("please input two integer:\n");
scanf("%d %d",&x,&y);
write(fda[1],&x,sizeof(int));
write(fda[1],&y,sizeof(int));
int result=0;
read(fdb[0],&result,sizeof(int));
printf("the result is : %d\n",result);
wait(0);
}
return 0;
}
實驗結果如下:
please input two integer:
5 7
the result is : 12