Linux下进程的通信方式: 有名管道(命名管道)
2016-05-20 22:00
615 查看
1.有名管道(命名管道)说明
命名管道通过路径名来标识,并且在文件系统中是可见的。
建立了管道之后,两个进程就可以把它当作普通文件一样进行读写操作。
命名管道(FIFO)突破了匿名管道只能用于具有亲缘关系的进程之间这种限制,它可使互不相关的两个进程实现彼此通信。
与匿名管道相同的是,FIFO也严格地遵循先进先出规则的,对管道及FIFO的读总是从开始处返回数据,对它们的写则把数据添加到末尾,它们不支持如lseek()等文件定位操作。
命名管道的创建:
可以使用函数mkfifo(),可以指定管道的路径、名字和打开的模式。
可以使用shell命令:mknod pipe_name p
命名管道的读写:
命名管道创建成功后,可以使用open()、read()和write()函数。
对于为读而打开的管道可在open()中设置O_RDONLY
对于为写而打开的管道可在open()中设置O_WRONLY
命名管道与普通文件的区别:
对普通文件的读写时不会出现阻塞问题
在命名管道的读写中却有阻塞的可能。
在open()函数中可以将模式设定为O_NONBLOCK以非阻塞方式打开。
(2)命名管道在阻塞与非阻塞打开方式下的读写问题
(1)对于读进程。
若该管道是阻塞打开,且当前FIFO内没有数据,则对读进程而言将一直阻塞到有数据写入。
若该管道是非阻塞打开,则不论FIFO内是否有数据,读进程都会立即执行读操作。即如果FIFO内没有数据,则读函数将立刻返回-1。
(2)对于写进程。
若该管道是阻塞打开,则写操作将一直阻塞到数据可以被写入。
若该管道是非阻塞打开而不能写入全部数据,则写操作进行部分写入或者调用失败。
mkfifo()函数格式:
3.使用实例
实例包含了两个程序
一个用于读管道(fifo_read.c)
创建管道
读管道
一个用于写管道(fifo_write.c)
由main()函数的参数传入用户需要写入的内容
将参数中的内容写入到管道
两个程序采用的均是阻塞式读写管道模式。
/* fifo_write.c */
/* fifo_rear.c */
运行:
首先启动读管道程序:读管道进程在建立管道之后就开始循环地从管道里读出内容,如果没有数据可读,则一直阻塞到写管道进程向管道写入数据。
再启动写管道程序:读进程能够从管道里读出用户的输入内容。
运行截图:
匿名管道相关知识,参见博客:http://blog.csdn.net/rl529014/article/details/51464363
命名管道通过路径名来标识,并且在文件系统中是可见的。
建立了管道之后,两个进程就可以把它当作普通文件一样进行读写操作。
命名管道(FIFO)突破了匿名管道只能用于具有亲缘关系的进程之间这种限制,它可使互不相关的两个进程实现彼此通信。
与匿名管道相同的是,FIFO也严格地遵循先进先出规则的,对管道及FIFO的读总是从开始处返回数据,对它们的写则把数据添加到末尾,它们不支持如lseek()等文件定位操作。
命名管道的创建:
可以使用函数mkfifo(),可以指定管道的路径、名字和打开的模式。
可以使用shell命令:mknod pipe_name p
命名管道的读写:
命名管道创建成功后,可以使用open()、read()和write()函数。
对于为读而打开的管道可在open()中设置O_RDONLY
对于为写而打开的管道可在open()中设置O_WRONLY
命名管道与普通文件的区别:
对普通文件的读写时不会出现阻塞问题
在命名管道的读写中却有阻塞的可能。
在open()函数中可以将模式设定为O_NONBLOCK以非阻塞方式打开。
(2)命名管道在阻塞与非阻塞打开方式下的读写问题
(1)对于读进程。
若该管道是阻塞打开,且当前FIFO内没有数据,则对读进程而言将一直阻塞到有数据写入。
若该管道是非阻塞打开,则不论FIFO内是否有数据,读进程都会立即执行读操作。即如果FIFO内没有数据,则读函数将立刻返回-1。
(2)对于写进程。
若该管道是阻塞打开,则写操作将一直阻塞到数据可以被写入。
若该管道是非阻塞打开而不能写入全部数据,则写操作进行部分写入或者调用失败。
mkfifo()函数格式:
3.使用实例
实例包含了两个程序
一个用于读管道(fifo_read.c)
创建管道
读管道
一个用于写管道(fifo_write.c)
由main()函数的参数传入用户需要写入的内容
将参数中的内容写入到管道
两个程序采用的均是阻塞式读写管道模式。
/* fifo_write.c */
#include <sys/types.h> #include <sys/stat.h> #include <errno.h> #include <fcntl.h> #include <stdio.h> #include <stdlib.h> #include <limits.h> #define MYFIFO "/tmp/myfifo" /* 有名管道文件名*/ #define MAX_BUFFER_SIZE PIPE_BUF /*常量PIPE_BUF 定义在于limits.h中*/ int main(int argc, char * argv[]) /*参数为即将写入的字符串*/ { int fd; char buff[MAX_BUFFER_SIZE]; int nwrite; if(argc <= 1) { printf("Usage: ./fifo_write string\n"); exit(1); } sscanf(argv[1], "%s", buff); /* 以只写阻塞方式打开FIFO管道 */ fd = open(MYFIFO, O_WRONLY); if (fd == -1) { printf("Open fifo file error\n"); exit(1); } /*向管道中写入字符串*/ if ((nwrite = write(fd, buff, MAX_BUFFER_SIZE)) > 0) { printf("Write '%s' to FIFO\n", buff); } close(fd); exit(0);4 }
/* fifo_rear.c */
#include <sys/types.h> #include <sys/stat.h> #include <errno.h> #include <fcntl.h> #include <stdio.h> #include <stdlib.h> #include <limits.h> #include <string.h> #define MYFIFO "/tmp/myfifo" /* 有名管道文件名*/ #define MAX_BUFFER_SIZE PIPE_BUF /*常量PIPE_BUF 定义在于limits.h中*/ int main() { char buff[MAX_BUFFER_SIZE]; int fd; int nread; /* 判断有名管道是否已存在,若尚未创建,则以相应的权限创建*/ if (access(MYFIFO, F_OK) == -1) { if ((mkfifo(MYFIFO, 0666) < 0) && (errno != EEXIST)) { printf("Cannot create fifo file\n"); exit(1); } } /* 以只读阻塞方式打开有名管道 */ fd = open(MYFIFO, O_RDONLY); if (fd == -1) { printf("Open fifo file error\n"); exit(1); } while (1) { memset(buff, 0, sizeof(buff)); if ((nread = read(fd, buff, MAX_BUFFER_SIZE)) > 0) { printf("Read '%s' from FIFO\n", buff); } } close(fd); exit(0); }
运行:
首先启动读管道程序:读管道进程在建立管道之后就开始循环地从管道里读出内容,如果没有数据可读,则一直阻塞到写管道进程向管道写入数据。
再启动写管道程序:读进程能够从管道里读出用户的输入内容。
运行截图:
匿名管道相关知识,参见博客:http://blog.csdn.net/rl529014/article/details/51464363
相关文章推荐
- Linux发行分支时间轴
- linux awk命令详解
- Linux中如何添加/删除FTP用户并设置权限(后续)
- 环境——LINUX上 JDK 的安装
- linux Cent-OS 安装中文输入法
- Linux下查看MySQL的安装路径
- 【Linux】多路复用之—select
- Linux is not Matrix——日志搜集平台ELK(II)
- Linux下的进程通信方式: 管道通信详解
- 在linux下安装android模拟器
- linux文件的三种时间
- Linux 命令行模式快捷键使用
- CentOS(5.8/6.4)linux生产环境若干优化实战
- 坑爹的一天——关于配置samba服务器(1)
- Linux源码环境下编译apk实践
- 从开始到结束,手把手教你使用busybox构建类嵌入式Linux系统
- Linux_2_Navigation
- Linux内核Makefile分析
- Linux文件权限
- selinux禁用后系统无法正常启动的问题