管道

管道是unix ipc的最古老形式,是一种在内存中的特殊文件,只能在具有公共祖先的进程之间使用(即父子进程,兄弟进程)。

管道由pipe函数创建

#include

int pipe(int fd[2])

fd[1]写,fd[0]读。

单个进程的管道几乎没有任何用处,通常,调用pipe的进程接着调用fork,这样就创建了父子进程间的管道。

#include #include#include#include

intmain()

{int fd[2];char buf[80];

pid_t pid;

pipe(fd);

pid=fork();if(pid>0)

{//父进程

printf("Father thread");char s[]="Hello";

write(fd[1],s,sizeof(s));

close(fd[0]);

close(fd[1]);

}else if(pid==0)

{

printf("Child Thread");

read(fd[0],buf,sizeof(buf));

printf("%s",buf);

close(fd[0]);

close(fd[1]);

}

waitpid(pid,NULL,0);//等待子进程结束

return 0;

}

输出结果:

Father thread

Child Thread

Hello

当管道的一端关闭时:

当读一个写端关闭的管道时,则认为已经读到了数据的末尾,读函数返回的读出字节数为0;

当写一个读端关闭的管道时,向管道中写入数据的进程将收到内核传来的SIFPIPE信号,应用程序可以处理该信号,也可以忽略(默认动作则            是应用程序终止)。

从管道中读取数据:

当管道的写端存在时,如果请求的字节数目大于PIPE_BUF,则返回管道中现有的数据字节数,如果请求的字节数目不大于PIPE_BUF,则返回管道中现有数据字节数(此时,管道中数据量小于请求的数据量);或者返回请求的字节数(此时,管道中数据量不小于请求的数据量)。注:PIPE_BUF在include/linux/limits.h中定义。

向管道中写入数据:

向管道中写入数据时,linux将不保证写入的原子性,管道缓冲区一有空闲区域,写进程就会试图向管道写入数据。如果读进程不读走管道缓冲区中的数据,那么写操作将一直阻塞。

管道因为没有名字所以只能用于具有亲缘关系的进程,而有名管道(FIFO)则克服了这个限制。

FIFO

创建函数如下

#include #include

int mkfifo(const char *pathname, mode_t mode);

第一个参数是一个普通的路径名,即为FIFO的名字。第二个参数设置权限,跟创建普通文件一样。

FIFO的读写也像普通文件一样,不过需要读写端都打开,具体规则如下:

当打开(open)时:

若没有设置O_NONBLOCK,只读open要阻塞到其它进程为写而打开FIFO。类似地,只写open要阻塞到其它进程为读而打开FIFO。

如果设置了O_NONBLOCK,则只读open立即返回,若没有其它进程为写而打开FIFO,则返回-1。

用FIFO模拟生产者消费者问题:

fifo2.cpp:

#include#include#include#include#include#include#include#include#include#include

#define FIFO "/tmp/myfifo"

#define BUF_SIZE PIPE_BUF

#define SEND_MAX (1024*1024*10)

using namespacestd;intmain()

{intpid,fifo_fd;intsend_num;char *buf[BUF_SIZE+1];if(-1 ==access(FIFO,F_OK))

{int res = mkfifo(FIFO,0777);if(res != 0)

{

fprintf(stderr,"can't create fifo in %s",FIFO);

exit(EXIT_FAILURE);

}

}

fifo_fd=open(FIFO,O_WRONLY);

printf("process %d open fifo %d",getpid(),fifo_fd);if(fifo_fd == -1)

exit(EXIT_FAILURE);intres;while(send_num

{

res=write(fifo_fd,buf,BUF_SIZE);if(res == -1)

{

cout<

exit(EXIT_FAILURE);

}

send_num+=res;

}return 0;

}

fifo3.cpp

#include#include#include#include#include#include#include#include#include#include

#define FIFO "/tmp/myfifo"

#define BUF_SIZE PIPE_BUF

#define SEND_MAX (1024*1024*10)

using namespacestd;intmain()

{intfifo_fd;intres;char buffer[BUF_SIZE+1];int read_num = 0;

fifo_fd=open(FIFO,O_RDONLY);

printf("process %d open FIFO %d",getpid(),fifo_fd);if(fifo_fd == -1)

exit(EXIT_FAILURE);do{

res=read(fifo_fd,buffer,BUF_SIZE);

read_num+=res;

}while(res>0);

close(fifo_fd);return 0;

}

结果如下:

可见读进程运行0.013s就读取了10m的数据,FIFO的效率还是很高的。

linux 管道 top,linux IPC总结——管道相关推荐

  1. linux操作系统进程间通信IPC之管道pipe及FIFO

    linux环境下,各进程相互独立,如果想要交换两个进程之间的数据,需要通过内核,在内存中提供一个缓存区,一个进程往缓存区中写数据,一个往缓存区读数据,内核提供的这种机制称为进程间通信(IPC),常见的 ...

  2. Linux系统编程---4(进程间通信IPC,管道)

    进程间通信目的 数据传输:一个进程需要将它的数据发送给另一个进程 资源共享:多个进程之间共享同样的资源. 通知事件:一个进程需要向另一个或一组进程发送消息,通知它(它们)发生了某种事件(如进程终止 时 ...

  3. linux进程管道通信缺点,Linux进程通信(IPC)的方式详解

    前言:Linux进程通信的方式 什么是进程通信?进程通信是指进程之间交换信息 进程通信方式共有6种: 管道(pipe),包括流管道(s_pipe)和有名管道(named pipe) 信号(signal ...

  4. Linux 3.进程间通信(IPC)(pipe 无名管道、mkfifo 有名管道、ftok、msgget、msgrcv、msgsnd、msgctl 消息队列)

    Linux 3.进程间通信(IPC) 进程间通信: 进程间方式: pipe 管道(无名管道) 头文件及原型 特点 pipe 示例 FIFO(有名管道) 管道文件的创建 mkfifo 头文件及原型 mk ...

  5. linux学习---linux基于文件的IPC(匿名管道pipe,命名管道mkfifo,普通文件,socket文件)

    常用的IPC分为两个类别,一是基于文件,而是基于内存 基于文件的分别有匿名管道,有名管道,普通的文件共享,socket文件 基于内存的有普通内存共享(本文章没有介绍),共享内存,共享信号量,消息队列 ...

  6. C++学习:第六章Linux高级编程 - (七)信号、sigqueue、sigaction、IPC、管道、匿名管道

    回顾: 1. 信号的作用 2. 理解信号: 软中断 可靠与不可靠信号 kill -l 3. 信号发送与注册 kill/raise alarm setitimer signal 4. 信号的屏蔽 sig ...

  7. linux命名管道通信方式图例,linux IPC 命名管道

    在前面一篇文章中[Linux进程间通信] - 匿名管道中,我们介绍了Linux/Unix系统中最古老的一种进程间通信方式 – 匿名管道.此外,我们也讲解了匿名管道的几个局限性,特别是匿名管道只能用于父 ...

  8. Linux进程全解12——lIPC机制之管道,SystemV IPC介绍

    以下内容源于朱有鹏<物联网大讲堂>的课程学习整理,如有侵权,请告知删除. 一.管道 1.管道(无名管道) (1)管道通信的原理:内核维护的一块内存,有读端和写端(管道是单向通信的): (2 ...

  9. linux进程间通信(IPC) ---无名管道

    管道概述 管道(pipe)又称无名管道 无名管道是一种特殊类型的文件,在应用层体现为两个打开的文件描述符 任何一个进程在创建的时候,系统都会,给它分配4G的虚拟内存,分为3G的用户空间和1G的内核空间 ...

最新文章

  1. 飞书,助力字节跳动高效成长的神秘引擎
  2. linux shell 脚本 查找文件,Linux Shell在目录下使用for循环结合if查找文件的巧用
  3. canvas之特丑时钟
  4. js正则--验证6-12位至少包含数字、小写字母和大些字母中至少两种字符,
  5. source insight 4.0 括号尾部提醒 及 常用快捷键
  6. Android中关于cpu/cpuset/schedtune的应用
  7. c++和python的区别、javascript_python和c++的区别
  8. 运行时动态调用子程序的例子
  9. How to use price determination in Quotation scenario
  10. nodejs + ts 配置
  11. centos 使用mutt发送邮件带附件
  12. poj 2503 Trie树
  13. BATJ一线大公司需要什么样的前端
  14. 动态规划 —— 树形 DP
  15. aes加密php源码,AES加解密类源码 · ThinkPHP5高阶实战教程 --诠释为API开发而生 · 看云...
  16. Sharding-Proxy安装_以及_sharding-proxy自动分表配置_Sharding-Sphere,Sharding-JDBC分布式_分库分表工作笔记019
  17. Python:获取命令行参数
  18. 加密解密技术基础、PKI及创建私有私有CA
  19. 【EMNLP2020】融合自训练和自监督方法的无监督文本顺滑研究
  20. July 16th 模拟赛C T3 圆周舞蹈 Solution

热门文章

  1. Android官方开发文档Training系列课程中文版:高效显示位图之加载大位图
  2. 一种基于伪标签半监督学习的小样本调制识别算法
  3. 2018acm-icpc宁夏邀请赛后记
  4. 设计法则 - 菲茨定律 (转)
  5. Beam概念学习系列之Pipeline 数据处理流水线
  6. magent + memcached部署过程
  7. 【推广】实用命令——tldr
  8. 3.1_ 4连续分配管理方式
  9. Hadoop伪分布式配置和搭建,hadoop单机安装,wordcount实例测试,hadoop安装java目录怎么找,问题及问题解决方法
  10. MySQL小问题:cant connect to MYSQL server on localhost