匿名管道只能于有血缘关系的进程,那么俩个没有任何关系的进程怎么通信,这就出现了命名管道来克服这一问题。命名管道是一个设备文件,只要访问该路径,就能通过这个共享资源来互相通信。FIFO是按照先进先出的原则工作的,第一个被写入的数据第一个先被读出。FIFO可被用于复制串行管道命令之间的输出流,于是也就不需要写数据到中间磁盘文件中(类似于使用管道以避免中间的磁盘文件);
命名管道的创建:
在shell下交互的创建命名管道
mkfifo命令创建管道:
mknod命令创建管道:
mknod [OPTION]... NAME TYPE [MAJOR MINOR]
option 有用的就是 -m 了
name 自定义
type 有 b (块设备),c(字符设备), p(管道)
在程序中使用系统函数建立命名管道
int mknod(const *filename, mode_t mode | S_IFIFO,(dev_t)0);
int mkfifo(const char *filename,mode_t mode)
mknod是比较老的函数,经常使用的是mkfifo(更加简单和规范)这里着重理解mkfifo这个函数。
filename为创建命名管道的全路径名,比如在当前路径下创建一个名为mypipe的管道,filename的值为"./mypipe"
创建管道:
#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
int main()
{
//umask(0);
if(mkfifo("./mypipe",S_IFIFO | 0666) < 0){
perror("mkfifo\n");
return -1;
}
return 0;
}
命名管道的使用:
与匿名管道的使用方法基本是相同的,不过使用命名管道时需要先调用open()打开;调用open时需要注意的是,调用open()打开命名管道的进程可能会被阻塞,但如果同时用读写方式(O_RDWR)打开,则不一定会导致阻塞;如果以只读方式(O_RDONLY)/(O_WRONLY)打开,则调用open()函数的进程会将被阻塞直到有写/读方式打开管道;
接下来会创建两个进程来验证没有血缘关系的两个进程通信;
进程1:
#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<string.h>
#include<unistd.h>
#include<fcntl.h>
int main() //client read
{
int fd = open("./mypipe",O_RDONLY);
if(fd < 0){
perror("open\n");
return -2;
}
char buf[100];
memset(buf,'\0',sizeof(buf));
while(1)
{
buf[0] = 0;
int ret = read(fd,buf,sizeof(buf));
if(ret <0){
perror("write error\n");
}
printf("sever:%s\n",buf);
}
close(fd);
return 0;
}
进程2:
#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<string.h>
#include<unistd.h>
#include<fcntl.h>
int main() //sever write
{
umask(0);
if(mkfifo("./mypipe",S_IFIFO | 0666) < 0){
perror("mkfifo\n");
return -1;
}
int fd = open("./mypipe",O_WRONLY);
if(fd < 0){
perror("open\n");
return -2;
}
char buf[100];
memset(buf,'\0',sizeof(buf));
while(1)
{
buf[0] = 0;
scanf("%s",buf);
if(write(fd,buf,sizeof(buf))<0){
perror("write error\n");
return -3;
}
//sleep(1);
}
close(fd);
return 0;
}
两进程通信:
命名管道是一个可见的文件,因此,它可以用于任何两个进程间进行通信。