Linux 进程间通信 有名管道(fifo)
有名管道特点:
1)无名管道只能用于具有亲缘关系的进程之间,这就限制了无名管道的使用范围
2)有名管道可以使互不相关的两个进程互相通信。
3)有名管道可以通过路径名来指出,并且在文件系统中可见,但内容存放在内存中。
4)进程通过文件IO来操作有名管道
5)有名管道遵循先进先出规则
6)不支持如lseek() 操作
注意:
以O_WRONLY打开管道,读阻塞
以O_RDWR打开管道,当管道中没有数据,读阻塞
//当进程用open打开有名管道用只读方式打开的话,则返回的文件描述符就代表管道的读端
创建有名管道:
int mkfifo(const char *filename, mode_t mode)
参数:filename 有名管道文件名(包括路径);mode 权限(读写0666)
成功返回 0 ,失败返回-1 并设置 errno 号 errno == EEXIST 时表示该有名管道已经存在
对有名管道的操作是通过文件IO 中的open read write 等操作的
例子:
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h> int main(int argc, const char *argv[])
{
char buf[] = {};
if(mkfifo("./fifo",) != ) //在当前路径下(运行程序所在的路径)创建有名管道,有名管道权限读写执行
{
if(errno == EEXIST) //当该有名管道存在时,提示下
{
printf("File exists\n");
}
else
{
perror("mkfifo fail ");
exit();
}
} int fd;
fd = open("./fifo",O_RDWR);//读写方式打开,使用文件IO 操作有名管道
if(fd < )
{
perror("open fifo fail: ");
exit();
}
write(fd,"",);
read(fd,buf,);
printf("%s\n",buf);
return ;
}
测试:
例子:通过有名管道让两个进程实现文件的复制
(1)读取文件写入有名管道中
/* 功能:实现在终端上输入获取文件名,读取文件内容,写到有名管道fifo中
* */
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
int main(int argc, const char *argv[])
{
char buf[] = {};
if(mkfifo("./fifo",) != ) //创建有名管道
{
if(errno == EEXIST)
{
printf("File exists\n");
}
else
{
perror("mkfifo fail ");
exit();
}
}
int fd_fifo,fd_file;
fd_fifo = open("./fifo",O_WRONLY);//只写方式打开,管道描述符
if(fd_fifo < )
{
perror("open fifo fail: ");
exit();
}
fd_file = open(argv[],O_RDONLY);//只读方式打开,源文件进行复制到管道中
if(fd_file < )
{
perror("open source fail ");
exit();
} //循环读取文件内容
ssize_t size;
while()
{
size = read(fd_file,buf,); //文件中读取数据,返回读取到多少数据
if(size <= )
{
break;
}
write(fd_fifo,buf,size);
}
close(fd_file);//关闭读的源文件
close(fd_fifo);
return ;
}
(2)读取有名管道中的数据,写入文件中实现复制
/* 功能:实现在有名管道中读取数据,写到文件中
* */
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
int main(int argc, const char *argv[])
{
char buf[] = {};
if(mkfifo("./fifo",) != ) //创建有名管道
{
if(errno == EEXIST) //有名管道存在的情况
{
printf("File exists\n");
}
else
{
perror("mkfifo fail ");
exit();
}
} int fd_fifo,fd_file; //此处fd_r是指有名管道,在有名管道中读取数据,写到文件中 fd_fifo = open("./fifo",O_RDONLY);//读方式打开
if(fd_fifo < )
{
perror("open fifo fail: ");
exit();
} fd_file = open(argv[],O_WRONLY|O_CREAT|O_TRUNC,);//把从有名管道中读取的数据,写到文件中,只读,没有创建,清空打开
if(fd_file < )
{
perror("fd_w open fail ");
exit();
}
//fifo 中循环读取数据,然后写到文件中
ssize_t size;
while()
{
size = read(fd_fifo,buf,); //读有名管道内容,返回读取多少个数据
if(size <= )
{
break;
}
write(fd_file,buf,size); //写入文件中
}
close(fd_fifo);
close(fd_file);
return ;
}
测试:此时要打开两个终端让两个进程进行通信,一个执行写操作,一个执行都操作(有名管道中有数据才可以读,且管道数据内容存在内存中)
Linux 进程间通信 有名管道(fifo)的更多相关文章
- linux进程间通信-有名管道(FIFO)
有名管道(FIFO) 命名管道也被称为FIFO文件,是一种特殊的文件.由于linux所有的事物都可以被视为文件,所以对命名管道的使用也就变得与文件操作非常统一. (1)创建命名管道 用如下两个函数中的 ...
- linux进程间通信--有名管道
有名管道 只有当一个库函数失败时,errno才会被设置.当函数成功运行时,errno的值不会被修改.这意味着我们不能通过测试errno的值来判断是否有错误存在.反之,只有当被调用的函数提示有错误发生时 ...
- Linux进程间通信之管道(pipe)、命名管道(FIFO)与信号(Signal)
整理自网络 Unix IPC包括:管道(pipe).命名管道(FIFO)与信号(Signal) 管道(pipe) 管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道 ...
- Linux 进程间通信之管道(pipe),(fifo)
无名管道(pipe) 管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道所具有的功能外,它还允许无亲缘关系进程间的通信: 定义函数: int pipe(int f ...
- Linux进程间通信之管道
1,进程间通信 (IPC ) Inter-Process Communication 比较好理解概念的就是进程间通信就是在不同进程之间传播或交换信息. 2,linux下IPC机制的分类:管道.信号.共 ...
- Linux 进程间通信(二) 管道
Linux 进程间通信-管道 进程是一个独立的资源分配单位,不同进程之间的资源是相互独立的,没有关联,不能在一个进程中直接访问另一个进程中的资源.但是,进程不是孤立的,不同的进程之间需要信息的交换以及 ...
- 有名管道FIFO
管道和FIFO的特征之一是它们的数据是一个字节流.这是UNIX的原生I/O模型.进程往其中写入的是字节流,系统不对它作解释. FIFO不存数据,只是通过它找到内核文件. 一.建立有名管道 1.命令mk ...
- Linux -- 进程间通信之管道
管道是 Linux 里的一种文件类型,同时也是 Linux 系统下进程间通信的一种方式 创建一个管道文件有两种方式: Shell 下命令 mkfifo + filename,即创建一个有名管道 ...
- linux进程间通信之管道篇
本文是对http://www.cnblogs.com/andtt/articles/2136279.html管道一节的进一步阐释和解释 1 管道 1.1 管道简介 管道是unix系统IPC的最古老的形 ...
随机推荐
- es+mongodb 整合
之前公司项目的数据都是从mysql查询,后面需求变更:同时技术上相应的也要改变策略,决定将mongodb和mysql的数据通过es建立索引来查询: 对于还没有接触或者真正了解es的可以先看一下相关Lu ...
- 正在从 Windows 应用商店下载... 无法从 Windows 应用商店下载。请检查网络连接。
手贱关掉了一下服务,再打开就是嘛
- Activiti获取当前活动(任务)的出口(动态生成提交按钮)
1.设置出口变量 当一个任务有一个或多个出口时,可以在出口连线出设置判断条件如图: 2.根据任务Id获取出口集合 public List<String> getOutGoingTransN ...
- Ubuntu16.04+cuda9.0安装教程
1.安装NVIDIA驱动 首先去官网(http://www.nvidia.cn/Download/index.aspx?lang=cn)查找适配自己电脑GPU的驱动,我的电脑驱动版本如下: 执行如下语 ...
- 请求一个url的全过程
最近在进行前端面试方面的一些准备,遇到了一个经典前端问题,一个url从输入到页面加载中间到底发生了什么,以前也认真想过这个问题,但是当时回答的都不全面,现在来好好总结一下: 总体来说分为以下六个步骤: ...
- shell变量替换扩展 字符串计数截取
- 让Nginx路径中的子目录匹配文件夹的另一种写法
其实相当于对路径做一种通配符,根据路径名访问相应的文件夹.直接看高潮部分如下.. location /static { root /var/www/usmt; index index.html boa ...
- No symbol table is loaded. Use the "file" command.
No symbol table is loaded. Use the "file" command. gdb 1. 首先使用gcc -g .c文件 -o 可执行文 ...
- “二维数组”?
首先说明C中是不存在所谓的"多维数组"的. int array[a][b]:声明的是一个有a个元素的数组,每个数组元素是一个含b个元素的数组. 上述声明正确的说法是,数组的数组. ...
- HIVE常用函数(1)聚合函数和序列函数
SUM--sum(汇总字段) over (partition by 分组字段 order by 排序字段) 如果不指定ROWS BETWEEN,默认为从起点到当前行;如果不指定ORDER BY,则将分 ...