管道是Linux支持的最初Unix IPC形式之一,具备如下特色:node
#include <unistd.h> int pipe(int fd[2])
该函数建立的管道的两端处于一个进程中间,在实际应用中没有太大意义,所以,一个进程在由pipe()建立管道后,通常再fork一个子进程,而后经过管道实现父子进程间的通讯(所以也不难推出,只要两个进程中存在亲缘关系,这里的亲缘关系指的是具备共同的祖先,均可以采用管道方式来进行通讯)。linux
管道两端可分别用描述字fd[0]以及fd[1]来描述,须要注意的是,管道的两端是固定了任务的。即一端只能用于读,由描述字fd[0]表示,称其为管道读端;另外一端则只能用于写,由描述字fd[1]来表示,称其为管道写端。若是试图从管道写端读取数据,或者向管道读端写入数据都将致使错误发生。通常文件的I/O函数均可以用于管道,如close、read、write等等。函数
从管道中读取数据:server
关于管道的读规则验证:blog
/************** * readtest.c * **************/ #include <unistd.h> #include <sys/types.h> #include <errno.h> main() { int pipe_fd[2]; pid_t pid; char r_buf[100]; char w_buf[4]; char* p_wbuf; int r_num; int cmd; memset(r_buf,0,sizeof(r_buf)); memset(w_buf,0,sizeof(r_buf)); p_wbuf=w_buf; if(pipe(pipe_fd)<0) { printf("pipe create error\n"); return -1; } if((pid=fork())==0) { printf("\n"); close(pipe_fd[1]); sleep(3);//确保父进程关闭写端 r_num=read(pipe_fd[0],r_buf,100); printf( "read num is %d the data read from the pipe is %d\n",r_num,atoi(r_buf)); close(pipe_fd[0]); exit(); } else if(pid>0) { close(pipe_fd[0]);//read strcpy(w_buf,"111"); if(write(pipe_fd[1],w_buf,4)!=-1) printf("parent write over\n"); close(pipe_fd[1]);//write printf("parent close fd[1] over\n"); sleep(10); } } /************************************************** * 程序输出结果: * parent write over * parent close fd[1] over * read num is 4 the data read from the pipe is 111 * 附加结论: * 管道写端关闭后,写入的数据将一直存在,直到读出为止. ****************************************************/
向管道中写入数据:进程
对管道的写规则的验证1:写端对读端存在的依赖性ip
#include <unistd.h> #include <sys/types.h> main() { int pipe_fd[2]; pid_t pid; char r_buf[4]; char* w_buf; int writenum; int cmd; memset(r_buf,0,sizeof(r_buf)); if(pipe(pipe_fd)<0) { printf("pipe create error\n"); return -1; } if((pid=fork())==0) { close(pipe_fd[0]); close(pipe_fd[1]); sleep(10); exit(); } else if(pid>0) { sleep(1); //等待子进程完成关闭读端的操做 close(pipe_fd[0]);//write w_buf="111"; if((writenum=write(pipe_fd[1],w_buf,4))==-1) printf("write to pipe error\n"); else printf("the bytes write to pipe is %d \n", writenum); close(pipe_fd[1]); } }
命名管道是一种特殊类型的文件,由于Linux中全部事物都是文件,它在文件系统中以文件名的形式存在。内存
在程序中,咱们能够使用两个不一样的函数调用来创建管道:get
#include <sys/types.h>cmd
#include <sys/stat.h>
int mkfifo(const char *filename, mode_t mode);
int mknode(const char *filename, mode_t mode | S_IFIFO, (dev_t) 0 );
1、实验:使用FIFO实现进程间通讯
两个独立的程序:
1. 生产者程序,它在须要时建立管道,而后尽量快地向管道中写入数据。
2. 消费者程序,它从FIFO中读取数据并丢弃它们。
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <fcntl.h> #include <limits.h> #include <unistd.h> #include <sys/types.h> #include <sys/stat.h> #define FIFO_NAME "../fifoname" #define BUFFER_SIZE PIPE_BUF #define TEN_MEG (1024 * 5 ) int main() { int pipe_fd; int res; int open_mode = O_WRONLY; int byte = 0; char buffer[BUFFER_SIZE + 1]; if (-1 == access(FIFO_NAME, F_OK)) { res = mkfifo(FIFO_NAME, 0777); if (res != 0) { printf("error create fifo pipe\n"); exit(EXIT_FAILURE); } } printf("Process %d opening FIFO O_WRONLY\n", getpid()); pipe_fd = open(FIFO_NAME, open_mode); printf("Process %d result %d\n", getpid(), pipe_fd); if (-1 != pipe_fd) { while (byte < TEN_MEG) { res = write(pipe_fd, buffer, BUFFER_SIZE); printf("write to pipe %d data! and buffsize if %d\n", res, BUFFER_SIZE); if (-1 == res) { printf("error write to pipe\n"); exit(EXIT_FAILURE); } byte += res; } close(pipe_fd); } else { exit(EXIT_FAILURE); } printf("Process %d finished\n", getpid()); }
消费者:
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <fcntl.h> #include <limits.h> #include <unistd.h> #include <sys/types.h> #include <sys/stat.h> #define FIFO_NAME "../fifoname" #define BUFFER_SIZE PIPE_BUF int main() { int pipe_fd; int open_mode = O_RDONLY; int res; char buffer[BUFFER_SIZE + 1]; int byte = 0; printf("Process %d opening FIFO O_RDONLY\n", getpid()); pipe_fd = open(FIFO_NAME, open_mode); printf("Process %d result %d\n", getpid(), pipe_fd); if (-1 != pipe_fd) { do { res = read(pipe_fd, buffer, BUFFER_SIZE); printf("read from server %d bytes data!\n", res); byte += res; sleep(1); }while (res > 0); close(pipe_fd); } else { exit(EXIT_FAILURE); } printf("Process %d finished, %d bytes read\n", getpid(), byte); return 0; }