Linux--进程间通信(管道及有名管道FIFO)(转)
一. 管道:
1.只能用于具有亲缘关系的进程之间的通信
2.半双工通信模式
3.一种特殊的文件,是一种只存在于内核中的读写函数
管道基于文件描述符,管道建立时,有两个文件描述符:
a. fd[0]: 固定用于读管道
b. fd[1]: 固定用于写管道
创建管道:pipe()
一般步骤:1. pipe()创建管道 2. fork()创建子进程 3. 子进程会继承父进程的管道
关闭管道:1. 逐个关闭文件描述符 2. close()
eg. 父子进程间的管道通信:父子进程对管道分别有自己的读写通道,把无关的读端或写段关闭。
View Code
1 #include <stdio.h> 2 #include <stdlib.h> 3 #include <string.h> 4 #include <sys/types.h> 5 #include <unistd.h> 6 7 #define MAX_DATA_LEN 256 8 #define DELAY_TIME 1 9 10 int main() { 11 pid_t pid; 12 char buf[MAX_DATA_LEN]; 13 const char *data="Pipe Test program"; 14 int real_read,real_write; 15 int pipe_fd[2]; 16 17 memset((void*)buf,0,sizeof(buf)); 18 19 if(pipe(pipe_fd)<0){ 20 perror("Pipe create error!\n"); 21 exit(1); 22 } 23 24 if ((pid=fork())<0) { 25 perror("Fork error!\n"); 26 exit(1); 27 } else if (pid==0) { 28 close(pipe_fd[1]); 29 sleep(DELAY_TIME*3); 30 31 if ((real_read=read(pipe_fd[0],buf,MAX_DATA_LEN))>0) { 32 printf("Child receive %d bytes from pipe: '%s'.\n",real_read,buf); 33 } 34 35 close(pipe_fd[0]); 36 exit(0); 37 } else { 38 close(pipe_fd[0]); 39 sleep(DELAY_TIME); 40 41 if ((real_write=write(pipe_fd[1],data,strlen(data)))>0) { 42 printf("Parent write %d bytes into pipe: '%s'.\n",real_write,data); 43 } 44 45 close(pipe_fd[1]); 46 waitpid(pid,NULL,0); 47 exit(0); 48 } 49 50 }
二. 有名管道FIFO
1. 使不相关的两个进程彼此通信:a. 通过路径名指出,在文件系统中可见
b. 管道建立后,两进程可按普通文件一样对其操作
2. FIFO遵循先进先出规则:a. 对管道读从开始处返回数据
b. 对管道写则把数据添加到末尾
c. 不支持如lseek()等文件定位操作
创建有名管道:mkfifo()
创建管道成功后,可使用open()、read()和write()等函数。
为读而打开的管道可在open()中设置O_RDONLY
为写而打开的管道可在open()中设置O_WRONLY
与普通文件不同的是阻塞问题
•普通文件的读写时不会出现阻塞问题
•在管道的读写中却有阻塞的可能,
•非阻塞标志:在open()函数中设定为O_NONBLOCK
l
阻塞打开与非阻塞打开
对于读进程
•若该管道是阻塞打开,且当前FIFO内没有数据,则对读进程而言将一直阻塞到有数据写入
•若该管道是非阻塞打开,则不论FIFO内是否有数据,读进程都会立即执行读操作。即如果FIFO内没有数据,则读函数将立刻返回0
对于写进程
•若该管道是阻塞打开,则写操作将一直阻塞到数据可以被写入
•若该管道是非阻塞打开而不能写入全部数据,则读操作进行部分写入或者调用失败