什么是管道
管道是单向的、先进先出的,它把一个进程的输出和另一个进程的输入连接在一起。一个进程(写进程)在管道的尾部写入数据,另一个进程(读进程)从管道的头部读出数据。
管道的分类
管道包括无名管道和命名管道两种,前者用于父进程和子进程间的通信,后者可用于运行于同一系统中的任意两个进程间的通信。
无名管道
无名管道的创建
无名管道由pipe( )函数创建:
int pipe(int filedis[2]);
当一个管道被创建时,它会创建两个文件描述符:filedis[0]用于读管道,filedis[1]用于写管道。
关闭管道
关闭管道只是将两个文件描述符关闭即可,可以使用普通的close函数逐个关闭。
无名管道读写
管道用于不同进程间通信。通常先创建一个管道,再通过fork函数创建一个子进程,该子进程会继承父进程创建的管道。
注意事项:必须在系统调用fork()前调用pipe(),否则子进程将不会继承文件描述符。,这样,就会创建两个管道,因为父子进程共享同一段代码段,都会各自调用pipe(),即建立两个管道,出现异常错误。
无名管道示例
1 #include <unistd.h> 2 #include <sys/types.h> 3 #include <sys/wait.h> 4 #include <errno.h> 5 #include <stdio.h> 6 #include <string.h> 7 #include <stdlib.h> 8 9 10 int main() 11 { 12 int pipe_fd[2]; 13 pid_t pid; 14 char buf_r[100]; 15 int r_num; 16 17 memset(buf_r,0,sizeof(buf_r)); 18 19 /*创建管道*/ 20 if(pipe(pipe_fd)<0) 21 { 22 printf("pipe create error "); 23 return -1; 24 } 25 26 /*创建子进程*/ 27 if((pid=fork())==0) //子进程执行序列 28 { 29 printf(" "); 30 close(pipe_fd[1]);//子进程先关闭了管道的写端 31 sleep(2); /*让父进程先运行,这样父进程先写子进程才有内容读*/ 32 if((r_num=read(pipe_fd[0],buf_r,100))>0) 33 { 34 printf("%d numbers read from the pipe is %s ",r_num,buf_r); 35 } 36 close(pipe_fd[0]); 37 exit(0); 38 } 39 else if(pid>0) //父进程执行序列 40 { 41 close(pipe_fd[0]); //父进程先关闭了管道的读端 42 if(write(pipe_fd[1],"Hello",5)!=-1) 43 printf("parent write1 Hello! "); 44 if(write(pipe_fd[1]," Pipe",5)!=-1) 45 printf("parent write2 Pipe! "); 46 close(pipe_fd[1]); 47 waitpid(pid,NULL,0); /*等待子进程结束*/ 48 exit(0); 49 } 50 return 0; 51 }
命名管道
命名管道和无名管道基本相同,但也有不同点:无名管道只能有父进程使用;但是通过命名管道,不相关的进程也能交换数据。
创建命名管道
#include <sys/types.h> #include <sys/stat.h> int mkfifo(const char *pathname,mode_t mode)
- pathname:FIFO文件名
- mode:属性(与文件操作相同)
一旦创建了一个FIFO,就可以用open打开它,一般的文件访问函数(close,read,write)都可以用于FIFO.
阻塞
当打开FIFO时,非阻塞标志(O_NONBLOCK)
将对以后的读写产生如下影响:
- 1 没有使用 O_NONBLOCK:访问要求无法满足时进程阻塞。如读取空的FIFO时,或者FIFO已满时。
- 2 使用O_NONBLOCK:访问要求无法满足时不阻塞,立即出错返回,error是ENXIO。
FIFO读进程
1 /********************************************************** 2 *程序要求: 使用mkfifo创建有名管道并实现两个进程之间的通讯。 3 *功能描述: 创建一个进程,并从已经建立好的有名管道中,读出事先写入的 4 **********************************************************/ 5 #include <sys/types.h> 6 #include <sys/stat.h> 7 #include <errno.h> 8 #include <fcntl.h> 9 #include <stdio.h> 10 #include <stdlib.h> 11 #include <string.h> 12 13 #define FIFO "/tmp/myfifo" 14 15 /* 16 * 程序入口 17 * */ 18 int main(int argc,char** argv) 19 { 20 char buf_r[100]; 21 int fd; 22 int nread; 23 24 printf("Preparing for reading bytes... "); 25 memset(buf_r,0,sizeof(buf_r)); 26 27 /* 打开管道 */ 28 fd=open(FIFO,O_RDONLY|O_NONBLOCK,0); 29 if(fd==-1) 30 { 31 perror("open"); 32 exit(1); 33 } 34 while(1) 35 { 36 memset(buf_r,0,sizeof(buf_r)); 37 38 if((nread=read(fd,buf_r,100))==-1) 39 { 40 if(errno==EAGAIN) 41 printf("no data yet "); 42 } 43 printf("read %s from FIFO ",buf_r); 44 sleep(1); 45 } 46 /*后面三句话是不会被运行到的,但不会影响程序运行的效果当程序在上面的死循环中执行时收到信号后会马上结束运行而没有执行后面的三句话*/ 47 close(fd); //关闭管道 48 pause(); /*暂停,等待信号*/ 49 unlink(FIFO); //删除文件 50 }
FIFO写进程
1 /********************************************************** 2 *程序要求: 使用mkfifo创建有名管道并实现两个进程之间的通讯。 3 *功能描述: 创建一个进程,并在其中创建一个有名管道,并向其写入数据。 4 **********************************************************/ 5 #include <sys/types.h> 6 #include <sys/stat.h> 7 #include <errno.h> 8 #include <fcntl.h> 9 #include <stdio.h> 10 #include <stdlib.h> 11 #include <string.h> 12 13 #define FIFO_SERVER "/tmp/myfifo" 14 15 /* 16 * 程序入口 17 * */ 18 int main(int argc,char** argv) 19 { 20 int fd; 21 char w_buf[100]; 22 int nwrite; 23 24 /*创建有名管道*/ 25 if((mkfifo(FIFO_SERVER,O_CREAT|O_EXCL|O_RDWR)<0)&&(errno!=EEXIST)) 26 { 27 printf("cannot create fifoserver "); 28 } 29 30 /*打开管道*/ 31 fd=open(FIFO_SERVER,O_WRONLY |O_NONBLOCK,0); 32 if(fd==-1) 33 { 34 perror("open"); 35 exit(1); 36 } 37 38 /*入参检测*/ 39 if(argc==1) 40 { 41 printf("Please send something "); 42 exit(-1); 43 } 44 strcpy(w_buf,argv[1]); 45 46 /* 向管道写入数据 */ 47 if((nwrite=write(fd,w_buf,100))==-1) 48 { 49 if(errno==EAGAIN) 50 printf("The FIFO has not been read yet.Please try later "); 51 } 52 else 53 { 54 printf("write %s to the FIFO ",w_buf); 55 } 56 close(fd); //关闭管道 57 return 0; 58 }