进程通信的实现方式有很多,本质都是在内核中 开辟一块缓冲区(可以是内核空间、磁盘文件映射缓冲、共享内存区等),通信双方, 一方把数据从用户空间写入内核缓冲区,另一方进程从缓冲区把数据读走。
1. 管道文件与普通文件的区别
2. 匿名管道
实例:利用匿名管道实现父子进程通信
(先建立pipe,然后fork将父进程克隆,但克隆的是管道的两个读写端口)
#include <unistd.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int main(){
int pipe_fd[2];
if(pipe(pipe_fd)<0){ //出错处理
printf("pipe create error\n");
return -1;
}//if
printf("pipe create success\n");
if (fork()>0) { //father
int r;
char buf[15+1];
printf("father Want to read from son\n"); ※Ⅰ
r=read(pipe_fd[0],buf,15); //从管道读端读取数据到buf
buf[r]=0;
printf("FATHER Got strings: %s\n",buf); ※Ⅱ
}
else { //son
const char *test="a test string !";
printf("Son is sleeping:\n"); ※Ⅲ
sleep(5);
printf("Son Write after sleep: %s\n",test); ※Ⅳ
write(pipe_fd[1],test,strlen(test)); //从管道写端写入test
}
close(pipe_fd[0]);
close(pipe_fd[1]);
} //main
运行结果如下:(注意四条输出语句的先后顺序)
Ⅰ和Ⅲ顺序随机,而Ⅳ输出语句必须在Ⅱ之前,必须先向管道写之后才能读。
3. 命名管道
(1)创建一个命名管道:
可以看到如果显示管道内容会停在这不动,只有按ctrl+c停止后才结束,所以此时我们开另一个终端,在那个终端向管道写内容,可以看到在此终端则能立即显示出已写入的内容(起到了同步的作用)。
(2)命名管道的特点总结:
- 允许无亲缘关系进程间的通信。
- FIFO以文件形式存在于文件系统中,但不通信的话文件内没有数据。
- 文件操作方式基于“先进先出”原理
- 提供灵活多样的同步机制