进程间通信
1. 进场间通信的方式
- 无名管道( pipe ):管道是一种半双工的通信方式,数据只能单向流动,而且只能在具有亲缘关系的进程间使用。进程的亲缘关系通常是指父子进程关系。
- 有名管道 (named pipe) : 有名管道也是半双工的通信方式,但是它允许无亲缘关系进程间的通信。
信号量( semophore ) : 信号量是一个计数器,可以用来控制多个进程对共享资源的访问。它常作为一种锁机制,防止某进程正在访问共享资源时,其他进程也访问该资源。因此,主要作为进程间以及同一进程内不同线程之间的同步手段。 - 消息队列( message queue ) : 消息队列是由消息的链表,存放在内核中并由消息队列标识符标识。消息队列克服了信号传递信息少、管道只能承载无格式字节流以及缓冲区大小受限等缺点。
信号 ( sinal ):信号是一种比较复杂的通信方式,用于通知接收进程某个事件已经发生。 - 共享内存( shared memory ) :共享内存就是映射一段能被其他进程所访问的内存,这段共享内存由一个进程创建,但多个进程都可以访问。共享内存是最快的 IPC 方式,它是针对其他进程间通信方式运行效率低而专门设计的。它往往与其他通信机制,如信号两,配合使用,来实现进程间的同步和通信。
- 套接字( socket ) : 套解口也是一种进程间通信机制,与其他通信机制不同的是,它可用于不同及其间的进程通信。
2. 无名管道
无名管道的特点:
- 只能用于有血缘关系的进程间通信
- 半双工的通信模式,具有固定的读写模式
- pipe看作特殊的文件来操作(又符合Linux一切皆文件的思想)
创建管道
pipe函数的原型是:
int pipe(int pipefd[2]);
它需要的头文件是:#include <unistd.h>
pipe函数的参数是一个有两个文件描述符(整型)元素的数组,pipe函数在数组中填入两个新的文件描述符后返回0,失败则返回1。
pipe()会建立管道,并将文件描述词由参数 filedes 数组返回。
pipefd[0]为管道里的读取端,所以pipe用read调用的
pipefd[1]则为管道的写入端。
Description:
- 建立一个管道
- 父进程写入一些内容 fork()
- 子进程读取一些内容,并打印出来
Thinking:
- pipe()建立管道
- fork()生成新进程
- 判断,如果是父,则写入管道内容pipefd[1]
- 判断,如果是子,则读出管道内容pipefd[0]
- 关闭文件描述符
无名管道的缺点: 只能实现父子进程(有亲缘关系进程)之间的通信。
3. 有名管道
正由于这无名管道的缺点,对无名管道进行改进:有名管道。
所谓的有名,即文件系统中存在这个一样文件节点,每一个文件节点都有一个inode号,而且这是一个特殊的文件类型:p管道类型。
mkfifo函数: 用来创建管道文件的节点,没有在内核中创建管道。只有通过open 函数打开这个文件时才会在内核空间创建管道。
头文件:
#include <sys/types.h>
#include <sys/stat.h>
函数:
int mkfifo(const char *filename, mode_t mode);
函数形式: int mkfifo(const char *filename,mode_t mode);
功能: 创建管道文件
参数: 管道文件文件名,权限,创建的文件权限仍然和umask有关系。
返回值: 创建成功返回0,创建失败返回-1。
4. 信号通信
信号通信的框架:
- 信号的发送(发送信号进程):kill raise alarm
- 信号的接收(接收信号进程) : pause() sleep while(1)
- 信号的处理(接收信号进程) : signal