无名管道通信
无名管道特点:
(1):只能用于具有亲缘关系的进程之间的通信。(父子进程或兄弟进程)
(2):是一个半双工的通信模式,具有固定的读端和写端。(fd[0]固定为读端,fd[1]固定为写端)
(3):管道也可以看成一种特殊的文件,但是它不属于任何文件系统,并且只存在于内存中。
无名管道的创建
用pipe()函数创建无名管道
所需的头文件:#include<unistd.h>
函数返回值:创建成功:0;创建失败:-1;
为何无名管道只能能够实现具有亲缘关系的进程间的通信:
因为:当父进程创建的无名管道后,创建进程时,子进程完全复制了父进程的所有内容,两者打开的文件描述符是一样的,所以子进程打开的管道就是父进程打开的管道。因此管道有名无名没有关系,故称为无名管道。因此只能实现亲缘关系进程间的通信。
实现父进程给子进程发送消息的完整程序代码如下:
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<stdlib.h>
#include<string.h>
int main()
{
int fd[2] = {-1,-1};
int ret = pipe(fd);//创建两个文件描述符
if(ret < 0)
{
perror("pipe error!\n");
return -1;
}
printf("pipe ok!\n");
pid_t pid = fork();//创建进程
if(pid < 0)
{
perror("fork error!");
close(fd[0]);
close(fd[1]);
return -1;
}
else if(pid > 0)
{
//进入父进程
while(1)
{
char buf[100] = {0};
printf("父进程给子进程发送消息:\n");
gets(buf);
int ret1 = write(fd[1],buf,sizeof(buf));//将父进程的数据写入fd[1]中。
if(ret1 < 0)
{
perror("write error!");
close(fd[0]);
close(fd[1]);
return 0;
}
sleep(1);
}
}
else
{
//进入子进程
while(1)
{
char buf[100] = {0};
int ret1 = read(fd[0],buf,sizeof(buf));//将管道中存储的数据读入子进程中。
if(ret1 < 0)
{
perror("read error!");
close(fd[0]);
close(fd[1]);
return -1;
}
else
{
printf("子进程接受到父进程发送的消息:\n");
puts(buf);
}
}
}
close(fd[0]);
close(fd[1]);
return 0;
}
程序运行结果: