前言

   进程是一个独立的资源管理单元,不同进程间的资源是独立的,不能在一个进程中访问另一个进程的用户空间和内存空间。但是,进程不是孤立的,不同进程之间需要信息的交互和状态的传递,因此需要进程间数据的传递、同步和异步的机制。

    当然,这些机制不能由哪一个进程进行直接管理,只能由操作系统来完成其管理和维护,Linux提供了大量的进程间通信机制,包括同一个主机下的不同进程和网络主机间的进程通信,如下图所示:

  • 同主机间的信息交互
  • 无名管道

    特点:多用于亲缘关系进程间通信,方向为单向;为阻塞读写;通信进程双方退出后自动消失

    问题:多进程用同一管道通信容易造成交叉读写的问题
  • 有名管道

    FIFO(First In First Out),方向为单向(双向需两个FIFO),以磁盘文件的方式存在;通信双方一方不存在则阻塞
  • 消息队列

    可用于同主机任意多进程的通信,但其可存放的数据有限,应用于少量的数据传递
  • 共享内存

    可实现同主机任意进程间大量数据的通信,但多进程对共享内存的访问存在着竞争
  • 同主机进程间同步机制:信号量(Semaphore)
  • 同主机进程间异步机制:信号(Signal)
  • 网络主机间数据交互:Socket(套接字)

1. 无名管道

1). 概念

    无名管道是一类特殊的文件,在内核中对应着一段特殊的内存空间,内核在这段内存空间中以循环队列的方式存储数据;

   无名管道的内核资源在通信双方退出后自动消失,无需人为回收;

   无名管道主要用于连通亲缘进程(父子进程),用于双方的快捷通信,通信为单向通信

2). 操作

1. 创建

pipe()

  • 头文件

        #include <unistd.h>
  • 函数原型

        int pipe(int pipedes[2])
  • 参数

pipedes[2]:存储文件描述符

  • 返回值

    成功:0

    失败:-1

2. 读写

读进程 写进程 管道无数据 管道有数据
阻塞√ 立刻返回0 返回数据
阻塞√ 读阻塞 读数据并返回
阻塞√ 收到SIGPIPE信号,write返回-1
阻塞√ 写入 若满,阻塞等待

3. 重定向

  所谓重定向,即关闭某个标准I/O设备(stdin(0), stdout(1), stderr(2)), 而将某一个普通的文件描述符设置为0/1/2.

  重定向,主要是采用 dup函数来实现的:

  • dup函数

  • 作用

    复制文件描述符

  • 头文件

        #include <unistd.h>
  • 函数原型

    函数 原型 描述
    dup int dup(int oldfd) 复制oldfd文件表项,并返回未用的最小值描述符
    dup2 int dup2(int oldfd, int newfd) 复制oldfd文件表项到newfd,newfd为用户自定义值,而后关闭oldfd
  • 参数

oldfd:源文件的描述符(要求该文件描述符必须有效,文件必须处于打开状态)

newfd:新的文件描述符,个人理解为oldfd的软链接

  • 返回值

    成功:最小值文件描述符(dup) / newfd的值(dup2)

    失败:-1

2. 有名管道

1). 概念

 又称FIFO(Fisrt In First out), 是磁盘上的一个特殊文件,没有在磁盘存储真正的信息,其所存储的信息在内存中

 通信双方进程结束后,自动丢弃FIFO中的信息,但其在磁盘的文件路径本身依然存在

2). 操作

1. 创建 - mkfifo()

  • 头文件

        #include <sys/stat.h>
  • 函数原型

        int mkfifo(const char *pathname, mode_t mode)
  • 参数

pathname: FIFO文件

mode: 创建文件的属性(权限)

  • 返回值

    成功:0

    失败:-1

2. 读写

读进程 写进程 FIFO无数据 FIFO有数据
阻塞 阻塞
阻塞 阻塞
写入 读出(未满,读写同时)
√x 即写中途退出,读直接返回0 same
√x 读中途退出,写返回SIGPIPE same

3. 补充

access函数

  • 作用

    检查用户(进程)对文件的权限

  • 头文件

        #include <unistd.h>
  • 函数原型

        int access(const char *pathname, int mode)
  • 参数

pathname: 即文件名

mode:测试的属性

mode description
F_OK 该文件是否存在
R_OK 可读
W_OK 可写
X_OK 可执行
bitwise 属性(读/写/执行等)
  • 返回值

    成功:0

    失败:-1

4. 代码示例

1). PIPE

/*
* Filename: pipe.c
*/ #include <stdio.h>
#include <unistd.h> //for pipe()
#include <string.h> //for memset()
#include <stdlib.h> //for exit() int main()
{
int fd[2];
char buf[20];
if(-1 == pipe(fd))
{
perror("pipe");
exit(EXIT_FAILURE);
} write(fd[1], "hello,world", 12);
memset(buf, '\0', sizeof(buf)); read(fd[0], buf, 12);
printf("The message is: %s\n", buf); return 0;
}

2). FIFO

程序的步骤:

  1. 创建一个新的fifo文件
  2. fifo_snd.c文件负责向fifo中写入数据
  3. fifo_rcv.c负责从fifo中读取数据并打印

1. fifo_snd.c:

/*
*File: fifo_send.c
*/ #include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/ipc.h>
#include <fcntl.h> #define FIFO "/tmp/my_fifo" int main()
{
char buf[] = "hello,world"; //`. check the fifo file existed or not
int ret;
ret = access(FIFO, F_OK);
if(ret == 0) //file /tmp/my_fifo existed
{
system("rm -rf /tmp/my_fifo");
} //2. creat a fifo file
if(-1 == mkfifo(FIFO, 0766))
{
perror("mkfifo");
exit(EXIT_FAILURE);
} //3.Open the fifo file
int fifo_fd;
fifo_fd = open(FIFO, O_WRONLY);
if(-1 == fifo_fd)
{
perror("open");
exit(EXIT_FAILURE); } //4. write the fifo file
int num = 0;
num = write(fifo_fd, buf, sizeof(buf));
if(num < sizeof(buf))
{
perror("write");
exit(EXIT_FAILURE);
} printf("write the message ok!\n"); close(fifo_fd); return 0;
}

2. fifo_rcv.c:

/*
*File: fifo_rcv.c
*/ #include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/ipc.h>
#include <fcntl.h> #define FIFO "/tmp/my_fifo" int main()
{
char buf[20] ;
memset(buf, '\0', sizeof(buf)); //`. check the fifo file existed or not
int ret;
ret = access(FIFO, F_OK);
if(ret != 0) //file /tmp/my_fifo existed
{
fprintf(stderr, "FIFO %s does not existed", FIFO);
exit(EXIT_FAILURE);
} //2.Open the fifo file
int fifo_fd;
fifo_fd = open(FIFO, O_RDONLY);
if(-1 == fifo_fd)
{
perror("open");
exit(EXIT_FAILURE); } //4. read the fifo file
int num = 0;
num = read(fifo_fd, buf, sizeof(buf)); printf("Read %d words: %s\n", num, buf); close(fifo_fd); return 0;
}

Linux进程间通信 -- 管道(pipe)的更多相关文章

  1. Linux进程间通信—管道

    Linux下的进程通信手段基本上是从UNIX平台上的进程通信手段继承而来的.而对UNIX发展做出重大贡献的两大主力AT&T的贝尔实验室及BSD(加州大学伯克利分校的伯克利软件发布中心)在进程间 ...

  2. Linux进程间通信:管道,信号量,消息队列,信号,共享内存,套接字

    Linux下的进程通信手段基本上是从UNIX平台上的进程通信手段继承而来的.而对UNIX发展做出重大贡献的两大主力AT&T的贝尔实验室及BSD(加州大学伯克利分校的伯克利软件发布中心)在进程间 ...

  3. linux中管道(pipe)一谈

    /*********************************************** 管道(pipe)是Linux上进程间通信的一种方式,其是半双工(数据流只能在一个方向上流动(还需要经过 ...

  4. Linux进程间通信-管道深入理解(转)

    原文地址:https://www.linuxidc.com/Linux/2018-04/151680.htm Linux进程通信系列文章将详细介绍各种通信方式的机制和区别 1.进程间通信 每个进程各自 ...

  5. Linux进程间通信---管道和有名管道

    一.管道 管道:管道是一种半双工的通信方式,数据只能单方向流动,而且只能在具有亲缘关系的进程间使用,因为管道 传递数据的单向性,管道又称为半双工管道.进程的亲缘关系通常是指父子进程关系. 管道的特点决 ...

  6. 详解linux进程间通信-管道 popen函数 dup2函数

    前言:进程之间交换信息的唯一方法是经由f o r k或e x e c传送打开文件,或通过文件系统.本章将说明进程之间相互通信的其他技术-I P C(InterProcess Communication ...

  7. linux 进程间通信之pipe

    在实际开发过程中,程序员必须让拥有依赖关系的进程集协调,这样才能达到进程的共同目标.  每个进程各自有不同的用户地址空间,任何一个进程的全局变量在另一个进程中都看不到,所以进程之间要交换数据必须通过内 ...

  8. Linux 进程间通信(管道、共享内存、消息队列、信号量)

           进程通信 : 不同进程之间传播或交换信息    为什么要进程通信呢? 协同运行,项目模块化 通信原理 : 给多个进程提供一个都能访问到的缓冲区. 根据使用场景,我们能划分为以下几种通信 ...

  9. linux进程间通信-管道

    一 管道的局限性 管道有两个局限性:(1)他是半双工(即数据只能在一个方向上流动).(2)它只能在具有公共祖先的进程之间使用.一个管道由一个进程创建,然后该 进程调用fork,此后父子进程之间就可该管 ...

随机推荐

  1. Checkpoint--相关问题

    Checkpoint是实例级别还是数据库级别? 答:数据库级别,在SQL Server关闭时,会对所有数据库逐一提交checkpoint 测试代码 USE DB0002 GO CHECKPOINT G ...

  2. C#全局键盘监听(Hook)

    一.为什么需要全局键盘监听? 在某些情况下应用程序需要实现快捷键执行特定功能,例如大家熟知的QQ截图功能Ctrl+Alt+A快捷键,只要QQ程序在运行(无论是拥有焦点还是处于后台运行状态),都可以按下 ...

  3. MVVM前端框架

    早开始接触MVVM框架的时候,是在学习WPF的时候,后面陆陆续续接触到了很多的前端JS框架,个人觉得大同小异,也没有去研究源代码,所以都停留在使用的阶段.当然对于我来说,使用这些JS框架,最关注的无非 ...

  4. VisualStudio神级插件Resharper的基本配置和使用技巧大全+Resharper性能优化

    所谓工欲善其事,必先利其器.尽管visual studio本身已经非常强大,但优秀的插件仍然可以帮开发者大大提高效率. ReSharper是一款由jetbrain开发的针对C#,VB.NET,ASP. ...

  5. 虚幻4随笔 三 从UE3到UE4

    笔者有幸参与过两个UE3项目,完全不同的使用方法,总共用了5.6年.引擎学习最好还是能参与项目,自己看的话往往容易纠结到一些细节上去,而引擎之所以是引擎,重要的恰恰是在容易被人忽视的工作流上.单从细节 ...

  6. foreach写失效的问题

    本文由作者张远道授权网易云社区发布. 坦白讲身为程序员,bug在所难免.有人讲,bug越多,说明程序员越伟大.这句话有它一定的道理. 因为从某方面讲,bug多了说明他的代码量也多. 言归正传,这里我记 ...

  7. iOS App的加固保护原理

    本文由  网易云发布. 本文从攻防原理层面解析了iOS APP的安全策略.iOS以高安全性著称,但它并非金刚不坏之身.对于信息安全而言,止大风于青萍之末是上上策,杭研深入各个细节的研发工作,正是网易产 ...

  8. FFmpeg软硬解和多线程解码

    一. AVCodecContext解码上下文 1.avcodec_register_all() : 注册所有的解码器 2.AVCodec *avcodec_find_decoder(enum AVCo ...

  9. day24 计算任意文件夹大小 , 校验大文件的一致性 , 发抢红包程序

    #!/usr/bin/env python# -*- coding:utf-8 -*- # 1.计算任意一个文件夹的大小(考虑绝对路径的问题)# 基础需求 这个文件夹中只有文件# 进阶需求 这个文件夹 ...

  10. node 无脑生成小程序二维码图

    RT 新建createwxaqrcode.js: const request = require('request') const fs = require('fs') // eg:生成购物车列表圆形 ...