【linux】进程通信

一 进程通信介绍

每个进程各自有不同的用户地址空间,任何一个进程的全局变量在另一个进程中都看不到,所以进
程之间要交换数据必须通过内核,在内核中开辟一块缓冲区,进程1把数据从用户空间拷到内核缓
冲区,进程2再从内核缓冲区把数据读走,内核提供的这种机制称为进程间通信
(IPC, InterProcess Communication) 。

1 进程通信目的

        1 数据传输:一个进程需要将它的数据发送给另一个进程

        2 资源共享:多个进程之间共享同样的资源。

        3 通知事件:一个进程需要向另一个或一组进程发送消息,通知它(它们)发生了某种事件(如进程终止 时要通知父进程)。

       4 进程控制:有些进程希望完全控制另一个进程的执行(如Debug进程),此时控制进程希望能够拦截另 一个进程的所有陷入和异常,并能够及时知道它的状态改变。

2 进程间通信发展

        1 管道

        2 System V进程间通信

        3 POSIX进程间通信

管道

        匿名管道pipe

        命名管道

System V IPC

        System V 消息队列

        System V 共享内存

        System V 信号量

POSIX IPC

        消息队列

        共享内存

        信号量

        互斥量

        条件变量

        读写锁

二 管道

什么是管道

        管道是Unix中最古老的进程间通信的形式。

        我们把从一个进程连接到另一个进程的一个数据流称为一个“管道”

匿名管道

示例代码:

用fork来共享管道原理

站在文件描述符角度-深度理解管道

站在内核角度-管道本质

管道读写规则

当没有数据可读时

        O_NONBLOCK disable:read调用阻塞,即进程暂停执行,一直等到有数据来到为止。         O_NONBLOCK enable:read调用返回-1,errno值为EAGAIN。

当管道满的时候

        O_NONBLOCK disable: write调用阻塞,直到有进程读走数据

         O_NONBLOCK enable:调用返回-1,errno值为EAGAIN

如果所有管道写端对应的文件描述符被关闭,则read返回0

如果所有管道读端对应的文件描述符被关闭,则write操作会产生信号SIGPIPE,进而可能导致write进程 退出

当要写入的数据量不大于PIPE_BUF时,linux将保证写入的原子性。

当要写入的数据量大于PIPE_BUF时,linux将不再保证写入的原子性。

管道特点

        只能用于具有共同祖先的进程(具有亲缘关系的进程)之间进行通信;通常,一个管道由一个进程创 建,然后该进程调用fork,此后父、子进程之间就可应用该管道。

        管道提供流式服务

        一般而言,进程退出,管道释放,所以管道的生命周期随进程

        一般而言,内核会对管道操作进行同步与互斥

        管道是半双工的,数据只能向一个方向流动;需要双方通信时,需要建立起两个管道

命名管道

        管道应用的一个限制就是只能在具有共同祖先(具有亲缘关系)的进程间通信。

        如果我们想在不相关的进程之间交换数据,可以使用FIFO文件来做这项工作,它经常被称为命名管道。

        命名管道是一种特殊类型的文件

创建一个命名管道

命名管道可以从命令行上创建,命令行方法是使用下面这个命令:

        $ mkfifo filename

命名管道也可以从程序里创建,相关函数有:

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

创建命名管道: $ mkfifo filename int mkfifo(const char *filename,mode_t mode); 

匿名管道与命名管道的区别

        匿名管道由pipe函数创建并打开。

        命名管道由mkfifo函数创建,打开用open

        FIFO(命名管道)与pipe(匿名管道)之间唯一的区别在它们创建与打开的方式不同,一但这些工作完 成之后,它们具有相同的语义。

命名管道的打开规则

如果当前打开操作是为读而打开FIFO时

        O_NONBLOCK disable:阻塞直到有相应进程为写而打开该FIFO

        O_NONBLOCK enable:立刻返回成功

如果当前打开操作是为写而打开FIFO时

        O_NONBLOCK disable:阻塞直到有相应进程为读而打开该FIFO

        O_NONBLOCK enable:立刻返回失败,错误码为ENXIO

例子 -用命名管道实现server&client通信

# ll
total 12
-rw-r--r--. 1 root root 46 Sep 18 22:37 clientPipe.c
-rw-r--r--. 1 root root 164 Sep 18 22:37 Makefile
-rw-r--r--. 1 root root 46 Sep 18 22:38 serverPipe.c
 
# cat Makefile
.PHONY:all
all:clientPipe serverPipe
clientPipe:clientPipe.c
 gcc -o $@ $^
serverPipe:serverPipe.c
 gcc -o $@ $^
 
.PHONY:clean
clean:
 rm -f clientPipe serverPipe

serverPipe.c

clientpipe.c

结果:

system V共享内存

共享内存区是最快的IPC形式。一旦这样的内存映射到共享它的进程的地址空间,这些进程间数据传递不再涉及到 内核,换句话说是进程不再通过执行进入内核的系统调用来传递彼此的数据

共享内存数据结构

共享内存函数

shmget函数

shmat函数

说明:

shmdt函数

shmctl函数

system V消息队列

        消息队列提供了一个从一个进程向另外一个进程发送一块数据的方法

        每个数据块都被认为是有一个类型,接收者进程接收的数据块可以有不同的类型值

特性方面

        IPC资源必须删除,否则不会自动清除,除非重启,所以system V IPC资源的生命周期随内核

system V信号量

信号量主要用于同步和互斥的,下面先来看看什么是同步和互斥。

进程互斥

        由于各进程要求共享资源,而且有些资源需要互斥使用,因此各进程间竞争使用这些资源,进程的这种 关系为进程的互斥

        系统中某些资源一次只允许一个进程使用,称这样的资源为临界资源或互斥资源。

        在进程中涉及到互斥资源的程序段叫临界区

特性方面

         IPC资源必须删除,否则不会自动清除,除非重启,所以system V IPC资源的生命周期随内核

  • 21
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值