共享内存可以说是最有用的进程间通信方式

上传人:cn****1 文档编号:497259431 上传时间:2023-11-16 格式:DOCX 页数:12 大小:23.11KB
返回 下载 相关 举报
共享内存可以说是最有用的进程间通信方式_第1页
第1页 / 共12页
共享内存可以说是最有用的进程间通信方式_第2页
第2页 / 共12页
共享内存可以说是最有用的进程间通信方式_第3页
第3页 / 共12页
共享内存可以说是最有用的进程间通信方式_第4页
第4页 / 共12页
共享内存可以说是最有用的进程间通信方式_第5页
第5页 / 共12页
点击查看更多>>
资源描述

《共享内存可以说是最有用的进程间通信方式》由会员分享,可在线阅读,更多相关《共享内存可以说是最有用的进程间通信方式(12页珍藏版)》请在金锄头文库上搜索。

1、共享内存可以说是最有用的进程间通信方式,也是最快的IPC形式。两个不同进程A、B共享内存的意思是:同一块物理内存被映射到进程A、B各自的进程地 址空间。进程A可以即时看到进程B对共享内存中数据的更新,反之亦然。由于多个进程 共享同一块内存区域,必然需要某种同步机制,互斥锁和信号量都可以。采用共享内存通信的一个显而易见的好处是效率高,因为进程可以直接读写内存,而不需要 任何数据的拷贝。对于像管道和消息队列等通信方式,则需要在内核和用户空间进行四次的数据拷贝,而共享 内存则只拷贝两次数据:一次从输入文件到共享内存区,另一次从共享内存区到输出文件。实际上,进程之间在共享内存时,并不总是读写少量数据后

2、就解除映射,有新的通信时,再 重新建立共享内存区域。而是保持共享区域,直到通信完毕为止,这样,数据内容一直保存 在共享内存中,并没有写回文件。共享内存中的内容往往是在解除映射时才写回文件的。因 此,采用共享内存的通信方式效率是非常高的。Linux的2.2.x内核支持多种共享内存方式,如mmap()系统调用,Posix共享内存,以及 系统V共享内存。linux发行版本如Redhat 8.0支持mmap()系统调用及系统V共享内存,但还没实现Posix 共享内存,本文将主要介绍mmap()系统调用及系统V共享内存API的原理及应用。一、内核怎样保证各个进程寻址到同一个共享内存区域的内存页面1、pa

3、ge cache 及 swap cache 中页面的区分:一个被访问文件的物理页面都驻留在page cache或swap cache 中, 一个页面的所有信息 由struct page 来描述。struct page中有一个域为指针 mapping,它指向一个struct address_space 类型结构。 page cache 或 swap cache 中的所有页面就是根据 address_space 结构以及一个偏移量来区分的。2、文件与address_space结构的对应:一个具体的文件在打开后,内核会在内存中为之建立一个 struct inode 结构,其中的 i_mapping

4、域指向一个 address_space结构。这样,一个文件就对应一个address_space 结构,一个address_space与一个偏移量能够确定一个page cache或swap cache中的 一个页面。因此,当要寻址某个数据时,很容易根据给定的文件及数据在文件内的偏移量而 找到相应的页面。3、进程调用mmap()时,只是在进程空间内新增了一块相应大小的缓冲区,并设置了相应 的访问标识,但并没有建立进程空间到物理页面的映射。因此,第一次访问该空间时,会引 发一个缺页异常。4、对于共享内存映射情况,缺页异常处理程序首先在 swap cache 中寻找目标页(符合 address_spa

5、ce 以及偏移量的物理页),如果找到,则直接返回地址;如果没有找到,则判 断该页是否在交换区(swap area),如果在,则执行一个换入操作;如果上述两种情况都不 满足,处理程序将分配新的物理页面,并把它插入到page cache中。进程最终将更新进程 页表。注:对于映射普通文件情况(非共享映射),缺页异常处理程序首先会在page cache中根 据address_space以及数据偏移量寻找相应的页面。如果没有找到,则说明文件数据还没 有读入内存,处理程序会从磁盘读入相应的页面,并返回相应地址,同时,进程页表也会更 新。5、所有进程在映射同一个共享内存区域时,情况都一样,在建立线性地址与物

6、理地址之间 的映射之后,不论进程各自的返回地址如何,实际访问的必然是同一个共享内存区域对应的 物理页面。注:一个共享内存区域可以看作是特殊文件系统shm中的一个文件,shm的安装点在交换 区上。上面涉及到了一些数据结构,围绕数据结构理解问题会容易一些。二、mmap()及其相关系统调用mmap()系统调用使得进程之间通过映射同一个普通文件实现共享内存。普通文件被映射到 进程地址空间后,进程可以向访问普通内存一样对文件进行访问,不必再调用read(),write() 等操作。注:实际上,mmap()系统调用并不是完全为了用于共享内存而设计的。它本身提供了不同 于一般对普通文件的访问方式,进程可以像

7、读写内存一样对普通文件的操作。而Posix或 系统V的共享内存IPC则纯粹用于共享目的,当然mmap()实现共享内存也是其主要应用 之一。1、mmap()系统调用形式如下:函数原型:void* mmap ( void * addr , size_t len , int prot , int flags , int fd , off_t offset )参数:addr 指定文件应被映射到进程空间的起始地址(一般被指定一个空指针,此时选择起始地 址的任务留给内核来完成);len是映射到调用进程地址空间的字节数,它从被映射文件开头offset个字节开始算起;prot参数指定共享内存的访问权限。可取如

8、下几个值的或:PROT_READ (可读),PROT_WRITE (可写),PROT_EXEC (可执行),PROT_NONE (不可访问);flags 由以下几个常值指定: MAP_SHARED , MAP_PRIVATE , MAP_FIXED,其中, MAP_SHARED , MAP_PRIVATE 必选其一,而 MAP_FIXED 则不推荐使用。fd为即将映射到进程空间的文件描述字,一般由open()返回,同时,fd可以指定为-1,此 时须指定flags参数中的MAP_ANON,表明进行的是匿名映射(不涉及具体的文件名,避 免了文件的创建及打开,很显然只能用于具有亲缘关系的进程间通信)

9、 ;offset参数一般设为0,表示从文件头开始offset处开始映射;返回值:如果调用成功,则返回文件映射到进程空间的地址,进程可直接操作起始地址为该值的有效 地址;如果调用失败,则返回MAP_FAILED (即(void *)=-1),并设置相应的err no;2、系统调用mmap()用于共享内存的两种方式:包含头文件:#include (1)使用普通文件提供的内存映射(适用于任何进程之间):注意:需要打开或创建一个文件,然后再调用 mmap();典型调用代码如下:cpp view plaincopy调用open(),以获取文件描述符fd=open(name, flag, mode);if

10、(fd0)/将打开的文件映射到进程空间中ptr=mmap(NULL, len , PROT_READ|PROT_WRITE, MAP_SHARED , fd , 0);通过mmap()实现共享内存的通信方式有许多特点和要注意的地方,我们将在范例中进行具 体说明。2)使用特殊文件提供匿名内存映射(适用于具有亲缘关系的进程之间) 由于父子进程特殊的亲缘关系,在父进程中先调用mmap(),然后调用fork()。那么在调用 fork()之后,子进程继承父进程匿名映射后的地址空间,同样也继承mmap()返回的地址, 这样,父子进程就可以通过映射区域进行通信了。注意:这里不是一般的继承关系。一般来说,子进

11、程单独维护从父进程继承下来的一些变量 而mmap()返回的地址,则由父子进程共同维护。对于具有亲缘关系的进程实现共享内存最好的方式应该是采用匿名内存映射的方式。此时 不必指定具体的文件,只要设置相应的标志即可。3、系统调用 munmap()该调用在进程地址空间中解除一个映射关系函数原型:int munmap( void * addr, size_t len )参数:addr是调用mmap()时返回的地址;len 是映射区的大小。返回值:成功则返回 0;失败则返回-1,并设置相应的err no (很有可能为EINVAL)当映射关系解除后,对原来映射地址的访问将导致段错误发生。4、系统调用 msy

12、nc()刷新进程对共享内容的改变到文件中;函数原型:int msync ( void * addr , size_t len, int flags)参数:addr是调用mmap()时返回的地址;len 是映射区的大小;flags 可以取值为 MS_ASYNC/ MS_SYNC/ MS_INVALIDATE,但不能同时为 MS_ASYNC 和 MS_SYNC其中,取值为MS_ASYNC (异步)时,调用会立即返回;取值为MS_SYNC (同步)时, 调用会等到更新结束之后返回;取MS_INVALIDATE(通知使用该共享区域的进程,数据已 经改变)时,在共享内容更改之后,使得文件的其他映射失效,

13、从而使得共享该文件的其他 进程去重新获取最新值;简而言之,就是MS_ASYNC和MS_SYNC是负责与磁盘文件的更新的,而MSNVALIDATE 是负责与其他共享该块内存的进程的同步的;一般说来,进程在映射空间的对共享内容的改变并不直接写回到磁盘文件中,往往在调用 mun map()后才执行该操作。可以通过调用msync()实现磁盘上文件内容与共享内存区的内容一致。三、mmap()范例下面将给出使用mmap()的两个范例:范例 1给出两个进程通过映射普通文件实现共享内存通信;范例 2给出父子进程通过匿名映射实现共享内存。系统调用mmap()有许多有趣的地方,下面是通过mmap ()映射普通文件

14、实现进程间的 通信的范例,我们通过该范例来说明mmap()实现共享内存的特点及注意事项。范例 1:两个进程通过映射普通文件实现共享内存通信范例 1 包含两个子程序:map_normalfilel.c 及 map_normalfile2.c0编译两个程序,可执行文件分别为map_normalfilel及map_normalfile2。两个程序通 过命令行参数指定同一个文件来实现共享内存方式的进程间通信。map_normalfile2 试图打开命令行参数指定的一个普通文件,把该文件映射到进程的地址 空间,并对映射后的地址空间进行写操作;map_normalfile1 把命令行参数指定的文件映射到进

15、程地址空间,然后对映射后的地址空 间执行读操作; 这样,两个进程通过命令行参数指定同一个文件来实现共享内存方式的进程间通信。下面是两个程序代码:cpp view plaincopy/*map_normalfile1.c*/#include #include #include #include #include #include #include typedef structchar name4;int age;people;int main(int argc, char* argv) / map a normal file as shared mem:char fileName32=0;strcpy(fileName, stuy1001593/temt);/创建一个文件,并打开int file_fd=open(fileName,O_CREAT|O_RDWR|O_TRUNC); if(file_fd=-1)pri ntf(”创建并打开文件失败n);exit(-1);/文件定位到offset处??lseek

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 学术论文 > 其它学术论文

电脑版 |金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号