【Linux 应用编程】进程管理 - 进程间通信IPC之共享内存 mmap
IPC(InterProcess Communication,进程间通信)是进程中的重要概念。Linux 进程之间常用的通信方式有:
- 文件:简单,低效,需要代码控制同步
- 管道:使用简单,默认阻塞
- 匿名管道 pipe:只存在于内核缓冲区,只能用于有血缘关系的进程
- 有名管道 FIFO:在文件系统中存在,可用于无血缘关系的进程
- 信号量:使用复杂,但开销小,操作系统本身支持信号量
- 内存映射区 mmap:进程有无血缘关系都可以
- 本地套接字 socket:稳定可靠
共享内存
通过共享内存,两个不相干的进程可以使用同一段逻辑内存。这块内存通常也是同一段物理内存。每个进程都把这段共享内存连接到自己的地址空间中,直接访问。
共享内存是进程间传递数据的常用方式,非常高效。
共享内存的使用
POSIX 共享内存区的使用包含四个步骤:
- 调用 shm_open 创建一个新的共享内存区对象(或打开一个以存在的共享内存区对象)
- 调用 mmap 把这个共享内存区映射到调用进程的地址空间
- 调用 munmap() 取消共享内存映射
- 调用 shm_unlink()函数删除共享内存段
在编译 POSIX 共享内存应用程序时需要加上 -lrt 参数。
shm_open 函数
shm_open() 函数用来打开或者创建一个共享内存区,两个进程可以通过给 shm_open() 函数传递相同的名字以达到操作同一共享内存的目的。
新建的共享内存区大小默认为 0,需要使用 ftruncate 函数设置大小才能使用。
函数原型:
#include <sys/mman.h>
#include <sys/stat.h>
#include <fcntl.h>
int shm_open(const char *name, int oflag, mode_t mode);
返回值:
成功返回创建或打开的共享内存描述符(与文件描述符相同作用),失败返回-1。
参数:
- name:创建的共享内存的名称,其它进程可以根据这个名称来打开共享内存
- oflag:以下值的或值:
- O_RDONLY:共享内存以只读方式打开;
- O_RDWR:共享内存以可读写方式打开;
- O_CREAT:共享内存不存在才创建;
- O_EXCL:如果指定了 O_CREAT,但共享内存已经存在时返回错误;
- O_TRUNC:如果共享内存已存在则将大小设置为 0;
- 参数 mode 只有指定 O_CREAT 才有效指出,指出共享内存的权限,与 open()函数
类似。
ftruncate 函数
新建的共享内存默认大小为 0,需要设置共享内存大小。ftruncate()函数可用来调整文件或者共享内存的大小。
函数原型:
#include <unistd.h>
#include <sys/types.h>
int ftruncate(int fd, off_t length);
函数成功返回 0,失败返回-1。
参数:
- fd:需要调整的共享内存或者文件
- length:大小
mmap 函数
共享内存创建或打开后,是不能在进程内直接使用的。需要先映射到当前进程的地址空间。
函数原型:
#include <sys/mman.h>
void *mmap(void *addr, size_t length, int prot, int flags,int fd, off_t offset);
参数:
- addr:指向映射存储区的起始地址,通常将其设置为 NULL,这表示由系统选择该
映射区的起始地址; - len:映射的字节数;
- prot:对映射存储区的保护要求,对指定映射存储区的保护要求不能超过文件 open
模式访问权限。它可以为以下值的或值:- PROT_READ:映射区可读);
- PROT_WRITE:映射区可写;
- PROT_EXEC:映射区可执行;
- PROT_NONE:映射区不可访问。
- flag:映射标志位,可为以下值的或值:
- MAP_FIXED:返回值必须等于 addr。因为这不利于可移植性,所以不鼓励使用此标志;
- MAP_SHARED:多个进程对同一个文件的映射是共享的,一个进程对映射的
内存做了修改,另一个进程也会看到这种变化;
MAP_PRIVATE:多个进程对同一个文件的映射不是共享的,一个进程对映射
的内存做了修改,另一个进程并不会看到这种变化。
- fd:要被映射的文件描述符或者共享内存描述符;
- offset:要映射字节在文件中的起始偏移量。
munmap 函数
函数原型:
#include <sys/mman.h>
int munmap(void *addr, size_t length);
函数成功返回 0,否则返回-1;
参数:
- addr 为 mmap()函数返回的地址
- length 是映射的字节数。取消映射后再对映射地址访问会导致调用进程收到 SIGSEGV 信号。
shm_unlink 函数
函数原型:
#include <sys/mman.h>
#include <sys/stat.h>
#include <fcntl.h>
int shm_unlink(const char *name);
函数成功返回 0,否则返回-1。
示例
两个没有血缘关系的进程,也可以使用共享内存进行进程间通信:
【Linux 应用编程】进程管理 - 进程间通信IPC之共享内存 mmap的更多相关文章
- 进程间通信IPC之--共享内存
每个进程各自有不同的用户地址空间,任何一个进 程的全局变量在另一个进程中都看不到,所以进程之间要交换数据必须通过内核,在内核中开辟一块缓冲 区,进程1把数据从用户空间拷到内核缓冲区,进程2再从内核缓冲 ...
- 进程间通信——IPC之共享内存
共享内存是三个IPC机制中的一个.它允许两个不相关的进程访问同一个逻辑内存.共享内存是在两个正在进行的进程之间传递数据的一种非常有效的方式. 大多数的共享内存的实现,都把由不同进程之间共享 ...
- Linux系统编程@进程管理(一)
课程目标: 构建一个基于主机系统的多客户即时通信/聊天室项目 涉及的理论知识 进程控制:僵尸进程/孤儿进程.进程控制.守护进程... 进程间通信:管道.命名管道.信号... 多线程编程: 锁.信号量. ...
- 【Linux 应用编程】进程管理 - 进程间通信IPC之管道 pipe 和 FIFO
IPC(InterProcess Communication,进程间通信)是进程中的重要概念.Linux 进程之间常用的通信方式有: 文件:简单,低效,需要代码控制同步 管道:使用简单,默认阻塞 匿名 ...
- Linux系统编程@进程管理(二)
1.创建守护进程(Deamon) 守护进程的概念与作用 后台服务程序 – 系统服务,进程名字往往以’d’结尾,生存周期比较长(系统装入时启动,关闭时候终止.系统装入两种启动方式:1从启动脚本.etc/ ...
- Linux系统编程@进程通信(一)
进程间通信概述 需要进程通信的原因: 数据传输 资源共享 通知事件 进程控制 Linux进程间通信(IPC)发展由来 Unix进程间通信 基于System V进程间通信(System V:UNIX系统 ...
- UNIX环境高级编程——进程管理和通信(总结)
进程管理与通信 进程的管理 进程和程序的区别: 进程: 程序的一次执行过程 动态过程,进程的状态属性会发生变化 程序:存放在磁盘上的指令.数据的有序集合 是个文件,可直观看到 程序program ...
- Linux学习之进程管理(十九)
Linux学习之进程管理 进程查看 查看系统中所有进程,使用BSD操作系统的格式 语法:ps aux 选项: a:显示所有前台进程 x:显示所有后台进程 u:显示这个进程是由哪个用户产生的 语法:ps ...
- 【Linux学习】Linux系统管理1—进程管理
Linux系统管理1-进程管理 一.Linux的三种进程 Linux包括3中不同类型的进程: 交互进程:由一个shell启动的进程.交互进程可以在前后台运行 批处理进程:该进程和终端无联系,是一个进程 ...
随机推荐
- 【转】Android编译系统详解(一)——build/envsetup.sh
出处 http://www.cloudchou.com/android/post-134.html 本文原创作者:Cloud Chou. 欢迎转载,请注明出处和本文链接 准备好编译环境后,编译Rom的 ...
- Service_Worker XSS
0x00 简介 Service Worker 是 Chrome 团队提出和力推的一个 WEB API,用于给 web 应用提供高级的可持续的后台处理能力.该 WEB API 标准起草于 2013 年, ...
- PIL 中的 Image 模块
转载:http://www.cnblogs.com/way_testlife/archive/2011/04/20/2022997.html PIL 中的 Image 模块 本文是节选自 PIL ...
- jquery 判断是否为空
jquery 判断是否为空 if(my_val == null || my_val == undefined || my_val==""){ console.log("我 ...
- VMware导入ova报错
报错如下: 此主机支持Intel VT-x,但Intel VT-x处于禁用状态. 解决方案如下: 联想E75主机,重启按F1进入BIOS Advanced—>CPU setup—>In ...
- 费用流 Dijkstra 原始对偶方法(primal-dual method)
简单叙述用Dijkstra求费用流 Dijkstra不能求有负权边的最短路. 类似于Johnson算法,我们也可以设计一个势函数,以满足在与原图等价的新图中的边权非负. 但是这个算法并不能处理有负圈的 ...
- 基于注解的IOC配置
1 明确 注解配置和XML配置要实现的功能都是一样的,都是要降低程序间的耦合.只是配置的形式不一样. 关于实际的开发中到底是使用XML还是注解,每家公司有着不同的习惯.具体问题具体分析. 2 环境搭建 ...
- Linux 软件的下载安装
一.Linux系统安装软件的方式有两种: 1.通过 Linux 资源服务(类似于APP Shop)直接安装 2.下载tar包,解压安装. 二.Linux 资源服务安装软件 1.提示:一般安装一个软 ...
- sqlserver表值函数调用方式
Connection conn = sqlServerManage.sqlServerConn(); Statement stmt; ResultSet rs; // 组装sql StringBuff ...
- 【NOIP2014模拟11.3】噪音
题目 FJ有M个牛棚,编号1至M,刚开始所有牛棚都是空的.FJ有N头牛,编号1至N,这N头牛按照编号从小到大依次排队走进牛棚,每一天只有一头奶牛走进牛棚.第i头奶牛选择走进第p[i]个牛棚.由于奶牛是 ...