Back to Blog

进程间通信系列(一)管道读写实例

#numbers#server#终端#null

一、管道读写注意点

1.只有在 管道 读端存在时,向管道写入才有意义;否则,会收到内核中的出错信号:SIFPIPE

2.向管道写入 数据 时不保证写入的原子性,管道缓冲区一有空闲区域,写进程就试图向其写入内容。若读进程不读取管道中的内容,则写进程会一直阻塞。

3.父子进程在运行时,它们的先后顺序得不到保证。因此在这里,为保证父进程关闭读描述符,可向子进程加入sleep(2)。

二、实例

1.无名管道

/*pipe_rw.c*/#include <unistd.h>#include <sys/types.h>#include <errno.h>#include <stdio.h>#include <stdlib.h> int main(){	int pipe_fd[2];	pid_t pid;	char buf_r[100];	char* p_wbuf;	int r_num;	memset(buf_r,0,sizeof(buf_r));	if(pipe(pipe_fd)<0)	{	printf("pipe create error\n");	return -1;	}	if((pid=fork())==0)    //若是子进程	{		printf("\n");		/*关闭子进程管道写端。睡眠2秒,确保父进程已相应地关闭了管道读端*/		close(pipe_fd[1]);		sleep(2);		/*子进程读取管道内容*/		if((r_num=read(pipe_fd[0],buf_r,100))>0){			printf(   "%d numbers read from the pipe is %s\n",r_num,buf_r);		}			/*关闭子进程读端*/		close(pipe_fd[0]);		exit(0);  	}	else if(pid>0)	{	/*关闭父进程读端*/		close(pipe_fd[0]);		/*分两次向管道写入数据*/		if(write(pipe_fd[1],"Hello",5)!=-1)			printf("parent write1 success!\n");		if(write(pipe_fd[1]," Pipe",5)!=-1)			printf("parent write2 success!\n");		/*关闭父进程写端并睡眠3秒,让子进程读数据*/		close(pipe_fd[1]);		sleep(3);		/*收集子进程退出信息*/		waitpid(pid,NULL,0);		exit(0);	}}

运行结果如下:

[root@localhost ipc]# ./pipe_rw parent write1 success!parent write2 success!10 numbers read from the pipe is Hello Pipe

2.有名管道

/*fifo_write.c*/#include <sys/types.h>#include <sys/stat.h>#include <errno.h>#include <fcntl.h>#include <stdio.h>#include <stdlib.h>#include <string.h>#define FIFO_SERVER "/tmp/myfifo" main(int argc,char** argv){	int fd;	char w_buf[100];	int nwrite;		if(fd==-1)		if(errno==ENXIO)			printf("open error; no reading process\n");	/*打开有名管道,并设置为非阻塞*/	fd=open(FIFO_SERVER,O_WRONLY|O_NONBLOCK,0);	if(argc==1)		printf("Please send something\n");	strcpy(w_buf,argv[1]);	/*向管道写入字符串*/	if((nwrite=write(fd,w_buf,100))==-1)	{		if(errno==EAGAIN)			printf("The FIFO has not been read yet.Please try later\n");	}	else 		printf("write %s to the FIFO\n",w_buf);}  /*fifo_read.c*/ #include <sys/types.h>#include <sys/stat.h>#include <errno.h>#include <fcntl.h>#include <stdio.h>#include <stdlib.h>#include <string.h>#define FIFO "/tmp/myfifo" main(int argc,char** argv){	char buf_r[100];	int  fd;	int  nread;		/*创建有名管道,并设置相应的权限*/	if((mkfifo(FIFO,O_CREAT|O_EXCL)<0)&&(errno!=EEXIST))		printf("cannot create fifoserver\n");	printf("Preparing for reading bytes...\n");		memset(buf_r,0,sizeof(buf_r));	/*打开有名管道,并设置非阻塞标志*/	fd=open(FIFO,O_RDONLY|O_NONBLOCK,0);	if(fd==-1)	{		perror("open");		exit(1);		}	while(1)	{		memset(buf_r,0,sizeof(buf_r));		/*读取管道中的字符串*/		if((nread=read(fd,buf_r,100))==-1){			if(errno==EAGAIN)				printf("no data yet\n");		}		printf("read %s from FIFO\n",buf_r);		sleep(1);	}		pause();	unlink(FIFO);}  

运行结果:

终端1:

[root@localhost ipc]# ./fifo_write 123123write 123123 to the FIFO

终端2:

[root@localhost ipc]# ./fifo_readPreparing for reading bytes...read  from FIFOread  from FIFOread  from FIFOread  from FIFOread  from FIFOread 123123 from FIFOread  from FIFO