1.管道(pipe)和有名管道(FIFO ):管道是单向的、先进先出的,它把一个进程的输出和另一个进程的输入连接在一起。一个进程(写进程)在管道的尾部写入数据,另一个进程(读进程)从管道的头部读出数据。
管道包括无名管道和有名管道两种,前者用于父进程和子进程间的通信,后者可用于运行于同一系统中的任意两个进程间的通信。
(1)无名管道由pipe()函数创建:int pipe(int filedis[2]);
当一个管道建立时,它会创建两个文件描述符:filedis[0]用于读管道,filedis[1]用于写管道
管道关闭:关闭管道只需将这两个文件描述符关闭即可,可以使用普遍的close函数逐个关闭。
例:管道通信
#include <unistd.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
int main()
{
int pipe_fd[2];
if(pipe(pipe_fd)<0)
{
printf("pipe create error\n");
return -1;
}
else
printf("pipe create success\n");
close(pipe_fd[0]);
close(pipe_fd[1]);
}
管道用于不同进程间同信。通常先创建一个管道,在通过fork函数创建一个子进程,该子进程会继承父进程所创建的管道。
例:管道通信
#include <unistd.h>
#include <sys/types.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
int main()
{
int pipe_fd[2];
pide_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 creat error\n”);
return -1;
}
/*创建子进程*/
if(pid=fork()==0)
{
printf(“\n”);
close(pipe_fd[1];
sleep(2);
if((r_num=read(pipe_fd[0],buf_r,1000))>0)
{
printf(“%d numbers read from the pipe is %s\n”,r_num,buf);
}
close(pipe_fd[0]);
exit(0);
}
else if(pid>0)
{
closr(pipe_fd[0]);
if(write(pipe_fd[1],”Hello”,5)!=-1)
printf(“parent writel Hello!\n”);
if(write(pipe_fd[1],”Ppe”,5)!=-1)
printf(“parent write2 Pipe!\n”);
close(pipe_fd[1]);
sleep(3);
waitpid (pid,NULL,0);/*等待子进程结束*/
exit(0);
}
}
(2)命名管道和无名管道基本相同,但也有不同点:无名管道只能由父子进程使用;但是通过命名管道,不相关的进程也能交换数据。
创建:
#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,将导致进程阻塞。
(2)使用O_NONBLOCK:访问要求无法满足时不阻塞,立即出错返回,errno是ENXIO。
例:(1)写进程
#include <sys/types.h>
#include <sys/stst.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;
/*打开管道*/
fd=open(FIFO_SERVER,O_WRONLY|O_NONBLOCK,0);
if(argc==1)
{
printf(“Please send something\n”);
exit(-1);
}
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);
}
(2)读进程
#include <sys/types.h>
#include <sys/stst.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)
{
printf(“open”);
exit(1);
}
while(1)
{
memset(buf_r,0,sizeof(buf_r));
if((nread=read(fdbuf_r,100))==-1)
{
if(errno==EAGAIN)
printf(“no data yet\n”);
}
printf(“read %s to the FIFO\n”, buf_r);
sleep(1);
}
pause();/*暂停,等待信号*/
unlink(FIFO);//删除文件
}