上一篇的管道通信可以看得出来,管道通信是单向的,只能一端写入,一端读取,有点类似通信中的单工通信,若要实现双工通信,可以采用两个单工通信。同样的道理,要实现管道双向通信,采用两个管道即可。
上图完成这样一项任务:父进程负责输入两个数据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