管道是单向的、先进先出的,它把一个进程的输出和还有一个进程的输入连接在一起。一个进程(写进程)在管道的尾部写入数据,还有一个进程(读进程)从管道的头部读出数据。数据被一个进程读出后,将被从管道中删除,其它读进程将不能再读到这些数据。管道提供了简单的流控制机制,进程试图读空管道时,进程将堵塞。相同,管道已经满时,进程再试图向管道写入数据。进程将堵塞。

管道包含无名管道和有名管道两种。无名管道仅仅能用于父进程和子进程间的通信,而有名管道能够用于同一系统中的随意两个进程间的通信。

  • 无名管道由pipe()函数创建
int pipe(int pipefd[2]);

#当一个无名管道建立时,它会创建两个文件描写叙述符:
#pipefd[0] 用于读管道,
#pipefd[1] 用于写管道。 #pipe()函数创建的管道默认是打开的.

演示样例程序:

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <errno.h>
#include <stdlib.h> int main(int argc, char** argv){
int pipe_fd[2];
pid_t pid;
if( pipe(pipe_fd) == -1){
fprintf(stderr, "%s\t%s\t%d\n",strerror(errno), __FILE__,__LINE__);
exit(-1);
}
if(( pid = fork()) == 0){
close(pipe_fd[0]);
write(pipe_fd[1],"hello ",6);
write(pipe_fd[1]," world",7); printf("Child process write done!\n");
close(pipe_fd[1]);
exit(0);
}else if( pid > 0){
close(pipe_fd[1]);
sleep(3);
char buf[20];
memset(buf,0,20);
read(pipe_fd[0],buf,20);
printf("Parent process in : %s\n",buf);
close(pipe_fd[0]);
} return 0;
} #对于无名管道的读、写、关闭操作,与普通文件的操作是一样的。
  • 有名管道的创建有两种方式

    1、使用shell命令,格式例如以下:
mkfifo [OPTION] FILENAME

2、在程序中调用mkfifo()函数

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

mkfifo()函数会根据參数 pathname 建立特殊的FIFO文件(有名管道),该文件必须不存在,而參数mode为该文件的权限。

mkfifo()建立的FIFO文件其它进程都能够用读写一般的文件方式存取。mkfifo()建立的FIFO文件默认是关闭的,当使用open()函数打开文件时须要注意和普通文件的差别:

1、不能以O_RDWR模式打开FIFO文件进行读写操作。这样做的行为是没有定义的。由于管道是单向的,假设须要在程序之间双向传递数据,使用一对有名管道就可以。

2、对标志位的 O_NONBLOCK 标志的使用方法。

O_RDONLY、O_WRONLY和O_NONBLOCK标志共同拥有四种合法的组合方式:

flags=O_RDONLY:open将会调用堵塞。除非有另外一个进程以写的方式打开同一个FIFO。否则一直等待。

flags=O_WRONLY:open将会调用堵塞。除非有另外一个进程以读的方式打开同一个FIFO。否则一直等待。

flags=O_RDONLY|O_NONBLOCK:假设此时没有其它进程以写的方式打开FIFO,此时open也会成功返回,此时FIFO被读打开,而不会返回错误。

flags=O_WRONLY|O_NONBLOCK:马上返回。假设此时没有其它进程以读的方式打开,open会失败打开。此时FIFO没有被打开。返回-1。

演示样例程序:

#fifo_read.c

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h> #define FIFO_FILE "/tmp/myfifo" int main(int argc, char *argv[]){
int fd ;
char buf[128];
if(access(FIFO_FILE,F_OK) != 0){
if(mkfifo(FIFO_FILE, 0644) == -1){
fprintf(stderr,"%s\t%s\t%d\n",strerror(errno), __FILE__,__LINE__);
return -1;
}
}
if( (fd = open(FIFO_FILE,O_RDONLY | O_NONBLOCK)) == -1){
fprintf(stderr,"%s\t%s\t%d\n",strerror(errno), __FILE__,__LINE__);
return -1;
} while(1){
memset(buf,0,sizeof(buf));
if(read(fd,buf,sizeof(buf)) == -1){
fprintf(stderr,"%s\t%s\t%d\n",strerror(errno), __FILE__,__LINE__);
return -1;
}
sleep(1);
printf("read is %s !\n",buf);
} unlink(FIFO_FILE);
return 0;
}
# fifo_write.c
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h> #define FIFO_FILE "/tmp/myfifo" int main(int argc, char *argv[]){
int fd ; if( (fd = open(FIFO_FILE,O_WRONLY | O_NONBLOCK)) == -1){
fprintf(stderr,"%s\t%s\t%d\n",strerror(errno), __FILE__,__LINE__);
return -1;
} if (write(fd,"Hello world",12) == -1){
fprintf(stderr,"%s\t%s\t%d\n",strerror(errno), __FILE__,__LINE__);
return -1;
} return 0;
}

Linux 进程通信之管道的更多相关文章

  1. Linux进程通信----匿名管道

    Linux进程通信中最为简单的方式是匿名管道 匿名管道的创建需要用到pipe函数,pipe函数参数为一个数组表示的文件描述字.这个数组有两个文件描 述字,第一个是用于读数据的文件描述符第二个是用于写数 ...

  2. linux进程通信之管道

    1.介绍: 1)同一主机: unix进程通信方式:无名管道,有名管道,信号 system v方式:信号量,消息队列,共享内存 2)网络通信:Socket,RPC 2.管道: 无名管道(PIPE):使用 ...

  3. linux 进程通信之 管道和FIFO

    进程间通信:IPC概念 IPC:Interprocess Communication,通过内核提供的缓冲区进行数据交换的机制. IPC通信的方式: pipe:管道(最简单) fifo:有名管道 mma ...

  4. Linux学习笔记(13)-进程通信|命名管道

    匿名管道只能在具有亲属关系的进程间通信,那么如果想要在不具有亲戚关系,想在陌生人之间通信,那又该怎么办呢? 别慌,Linux身为世界上*强大的操作系统,当然提供了这种机制,那便是命名管道-- 所谓命名 ...

  5. linux 进程通信 管道

    1. 管道概述及相关API应用 1.1 管道相关的关键概念 管道是Linux支持的最初Unix IPC形式之一,具有以下特点: 管道是半双工的,数据只能向一个方向流动:需要双方通信时,需要建立起两个管 ...

  6. Linux下进程通信之管道

    每个进程各自有不同的用户地址空间,任何一个进程的全局变量在另一个进程中都看不到,所以进程之间要交换数据必须通过内核,在内核中开辟一块缓冲区,进程1把数据从用户空间拷到内核缓冲区,进程2再从内核缓冲区把 ...

  7. Linux进程通信之匿名管道

    进程间的通信方式 进程间的通信方式包括,管道.共享内存.信号.信号量.消息队列.套接字. 进程间通信的目的 进程间通信的主要目的是:数据传输.数据共享.事件通知.资源共享.进程控制等. 进程间通信之管 ...

  8. linux下的进程通信之管道与FIFO

    概念:管道是由内核管理的一个缓冲区,相当于我们放入内存中的一个纸条.管道的一端连接一个进程的输出.这个进程会向管道中放入信息.管道的另一端连接一个进程的输入,这个进程取出被放入管道的信息. 优点:不需 ...

  9. 进程通信类型 管道是Linux支持的最初Unix IPC形式之一

    管道 Linux环境进程间通信(一) https://www.ibm.com/developerworks/cn/linux/l-ipc/part1/index.html 管道及有名管道 郑彦兴200 ...

随机推荐

  1. BZOJ1685: [Usaco2005 Oct]Allowance 津贴

    [传送门:BZOJ1685] 简要题意: 贝西工作勤勤恳恳,她每月向约翰索要C 元钱作为工资.约翰手上有不少钱,他一共有N 种面 额的钞票.第i 种钞票的面额记作Vi,约翰有Ki 张.钞票的面额设定是 ...

  2. jquery<=======>js实现方法

    http://haorooms.com/post/js_shixian_jquery

  3. 在Windows下如何创建指定的虚拟环境

    前几天给大家分享了如何在默认的情况下创建虚拟环境,没来得及上车的伙伴,可以戳这篇文章:在Windows下如何创建虚拟环境(默认情况下).今天小编给大家分享一下,如何创建的指定的Python环境. 创建 ...

  4. python 多线程学习小记

    python对于thread的管理中有两个函数:join和setDaemon setDaemon:如果在程序中将子线程设置为守护线程,则该子线程会在主线程结束时自动退出,设置方式为thread.set ...

  5. C++11 volatile 类型

    volatile作用: 作为指令关键字,确保本条指令不会受到编译器的优化而省略,而且要求每次直接读值. 定义: volatile int nTest; volatile关键字是一种类型修饰符,用它声明 ...

  6. CentOS下部署巡风步骤详解

    本博客已经迁移到新的网站,www.je2se.com,请大家移步关注,互相交流,共同成长 巡风Centos 6.5部署指南 基础环境要求: Python2.7+ 安装Centos相关依赖 # Cent ...

  7. js---03属性操作

    <!DOCTYPE HTML> <html> <head> <meta http-equiv="Content-Type" content ...

  8. Python 数据结构与算法 —— 哈弗曼树

    1. 从扩充二叉树到哈弗曼树 扩充二叉树:对二叉树 T,加入足够多的新叶节点(而不是任意),使 T 的原有结点都变成度数为 2 的分支节点,得到的二叉树称为 T 的扩充二叉树. 对于扩充二叉树而言, ...

  9. js中常用的对象—Array的属性和方法

    今天说一下,js中常用的内置对象——Array对象 Array常用属性: length prototype :给系统对象添加属性和方法 Array常用方法: Array.prototype.sum = ...

  10. Elasticsearch之需要注意的问题(es和jdk版本)

    (1)在使用java代码操作es集群的时候 要保证本地使用的es的版本和集群上es的版本保持一致.  (2)保证集群中每个节点的JDK版本和es基本配置一致 这个很简单,不多说.  (3)es集群中j ...