概述

url=MdyPihmS_tWLwgWL5CMzaTrwDFHu6euAJJUAjKvlzbJmRw7RfhmkBWwAloo7Y65hLY-kQdHsbqWYP2wc2fk8yq">共享内存是进程间通信中最简单的方式之中的一个。共享内存同意两个或很多其他进程訪问同一块内存,就如同 malloc() 函数向不同进程返回了指向同一个物理内存区域的指针。

当一个进程改变了这块地址中的内容的时候,其他进程都会察觉到这个更改。

共享内存的特点:

1)共享内存是进程间共享数据的一种最快的方法。

一个进程向共享的内存区域写入了数据,共享这个内存区域的全部进程就能够立马看到当中的内容。

2)使用共享内存要注意的是多个进程之间对一个给定存储区訪问的相互排斥。

若一个进程正在向共享内存区写数据,则在它做完这一步操作前,别的进程不应当去读、写这些数据。

经常使用函数

1)创建共享内存

所需头文件:

#include <sys/ipc.h>

#include <sys/shm.h>

int shmget(key_t key, size_t size,int shmflg);

功能:

创建或打开一块共享内存区。

參数:

key:进程间通信键值。ftok() 的返回值。

size:该共享存储段的长度(字节)。

shmflg:标识函数的行为及共享内存的权限。其取值例如以下:

IPC_CREAT:假设不存在就创建

IPC_EXCL:  假设已经存在则返回失败

位或权限位:共享内存位或权限位后能够设置共享内存的訪问权限,格式和 open() 函数的 mode_t 一样(open() 的使用请点此链接),但可运行权限未使用。

返回值:

成功:共享内存标识符。

失败:-1。

演示样例代码例如以下:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <unistd.h>
  5. #include <sys/types.h>
  6. #include <sys/ipc.h>
  7. #include <sys/shm.h>
  8.  
  9. #define BUFSZ 1024
  10.  
  11. int main(int argc, char *argv[])
  12. {
  13. int shmid;
  14. key_t key;
  15.  
  16. key = ftok("./", 2015);
  17. if(key == -1)
  18. {
  19. perror("ftok");
  20. }
  21.  
  22. //创建共享内存
  23. shmid = shmget(key, BUFSZ, IPC_CREAT|0666);
  24. if(shmid < 0)
  25. {
  26. perror("shmget");
  27. exit(-1);
  28. }
  29.  
  30. return 0;
  31. }

执行结果例如以下:

2)共享内存映射

所需头文件:

#include <sys/types.h>

#include <sys/shm.h>

void *shmat(int shmid, const void *shmaddr, int shmflg);

功能:

将一个共享内存段映射到调用进程的数据段中。简单来理解,让进程和共享内存建立一种联系,让进程某个指针指向此共享内存。

參数:

shmid:共享内存标识符。shmget() 的返回值。

shmaddr:共享内存映射地址(若为 NULL 则由系统自己主动指定)。推荐使用 NULL

shmflg:共享内存段的訪问权限和映射条件( 通常为 0 ),详细取值例如以下:

0:共享内存具有可读可写权限。

SHM_RDONLY:仅仅读。

SHM_RND:(shmaddr 非空时才有效)

返回值:

成功:共享内存段映射地址( 相当于这个指针就指向此共享内存 )

失败:-1

3)解除共享内存映射

所需头文件:

#include <sys/types.h>

#include <sys/shm.h>

int shmdt(const void *shmaddr);

功能:

将共享内存和当前进程分离( 不过断开联系并不删除共享内存。相当于让之前的指向此共享内存的指针,不再指向)。

參数:

shmaddr:共享内存映射地址。

返回值:

成功:0

失败:-1

4)共享内存控制

所需的头文件:

#include <sys/ipc.h>

#include <sys/shm.h>

int shmctl(int shmid, int cmd, struct shmid_ds *buf);

功能:

共享内存属性的控制。

參数:

shmid:共享内存标识符。

cmd:函数功能的控制,其取值例如以下:

IPC_RMID:删除。(经常使用
)

IPC_SET:设置 shmid_ds 參数,相当于把共享内存原来的属性值替换为 buf 里的属性值。

IPC_STAT:保存 shmid_ds 參数,把共享内存原来的属性值备份到 buf 里。

SHM_LOCK:锁定共享内存段( 超级用户 )。

SHM_UNLOCK:解锁共享内存段。

SHM_LOCK 用于锁定内存,禁止内存交换。

并不代表共享内存被锁定后禁止其他进程訪问。其真正的意义是:被锁定的内存不同意被交换到虚拟内存中。这样做的优势在于让共享内存一直处于内存中,从而提高程序性能。

buf:shmid_ds 数据类型的地址(详细类型请点此链接 ),用来存放或改动共享内存的属性。

返回值:

成功:0

失败:-1

实战演示样例

接下来我们做这么一个样例:创建两个进程,在 A 进程中创建一个共享内存,并向其写入数据。通过 B 进程从共享内存中读取数据。

写端代码例如以下:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <unistd.h>
  5. #include <sys/types.h>
  6. #include <sys/ipc.h>
  7. #include <sys/shm.h>
  8.  
  9. #define BUFSZ 512
  10.  
  11. int main(int argc, char *argv[])
  12. {
  13. int shmid;
  14. int ret;
  15. key_t key;
  16. char *shmadd;
  17.  
  18. //创建key值
  19. key = ftok("../", 2015);
  20. if(key == -1)
  21. {
  22. perror("ftok");
  23. }
  24.  
  25. //创建共享内存
  26. shmid = shmget(key, BUFSZ, IPC_CREAT|0666);
  27. if(shmid < 0)
  28. {
  29. perror("shmget");
  30. exit(-1);
  31. }
  32.  
  33. //映射
  34. shmadd = shmat(shmid, NULL, 0);
  35. if(shmadd < 0)
  36. {
  37. perror("shmat");
  38. _exit(-1);
  39. }
  40.  
  41. //拷贝数据至共享内存区
  42. printf("copy data to shared-memory\n");
  43. bzero(shmadd, BUFSZ); // 共享内存清空
  44. strcpy(shmadd, "how are you, mike\n");
  45.  
  46. return 0;
  47. }

读端代码例如以下:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. #include <unistd.h>
  5. #include <sys/types.h>
  6. #include <sys/ipc.h>
  7. #include <sys/shm.h>
  8.  
  9. #define BUFSZ 512
  10.  
  11. int main(int argc, char *argv[])
  12. {
  13. int shmid;
  14. int ret;
  15. key_t key;
  16. char *shmadd;
  17.  
  18. //创建key值
  19. key = ftok("../", 2015);
  20. if(key == -1)
  21. {
  22. perror("ftok");
  23. }
  24.  
  25. system("ipcs -m"); //查看共享内存
  26.  
  27. //打开共享内存
  28. shmid = shmget(key, BUFSZ, IPC_CREAT|0666);
  29. if(shmid < 0)
  30. {
  31. perror("shmget");
  32. exit(-1);
  33. }
  34.  
  35. //映射
  36. shmadd = shmat(shmid, NULL, 0);
  37. if(shmadd < 0)
  38. {
  39. perror("shmat");
  40. exit(-1);
  41. }
  42.  
  43. //读共享内存区数据
  44. printf("data = [%s]\n", shmadd);
  45.  
  46. //分离共享内存和当前进程
  47. ret = shmdt(shmadd);
  48. if(ret < 0)
  49. {
  50. perror("shmdt");
  51. exit(1);
  52. }
  53. else
  54. {
  55. printf("deleted shared-memory\n");
  56. }
  57.  
  58. //删除共享内存
  59. shmctl(shmid, IPC_RMID, NULL);
  60.  
  61. system("ipcs -m"); //查看共享内存
  62.  
  63. return 0;
  64. }

执行结果例如以下:

本教程演示样例代码下载请点此处。

Linux系统编程——进程间通信:共享内存的更多相关文章

  1. linux系统编程--进程间通信

    IPC方法 Linux环境下,进程地址空间相互独立,每个进程各自有不同的用户地址空间.任何一个进程的全局变量在另一个进程中都看不到,所以进程和进程之间不能相互访问, 要交换数据必须通过内核,在内核中开 ...

  2. Linux系统编程——进程间通信(System V IPC 对象)

    基本查看命令 ipcs  -m查看共享内存        ipcs -s查看信号量        ipcs -q查看消息队列 ipcrm  -m  id 删除共享内存   -M+key值 ipcrm ...

  3. Linux系统编程——进程间通信:信号中断处理

    什么是信号? 信号是 Linux 进程间通信的最古老的方式.信号是url=474nN303T2Oe2ehYZjkrggeXCaJPDSrmM5Unoh4TTuty4wSgS0nl4-vl43AGMFb ...

  4. linux网络编程之共享内存介绍

    今天是个好日子,洋人之节乃全球同庆,圣诞一来感觉就要过年了,不过今晚心情有点打折扣,给心爱的人打电话没有打通,本想在平安夜送上快乐的祝福给她,糟糕的心情让自己好像泄了气的皮球一样,无精打彩,心情灰暗, ...

  5. Linux环境编程之共享内存区(一):共享内存区简单介绍

    共享内存区是可用IPC形式中最快的.一旦内存区映射到共享它的进程的地址空间,进程间数据的传递就不再涉及内核.然而往该共享内存区存放信息或从中取走信息的进程间通常须要某种形式的同步.不再涉及内核是指:进 ...

  6. Linux系统编程——进程间通信(一)

    基本操作命令: ps -ajx/-aux/-ef 查看进程间状态/的相互关系 top 动态显示系统中的进程 nice 按照指定的优先级运行 /renice 改变正在运行的进程的优先级 kill -9杀 ...

  7. Linux系统编程——进程间通信:管道(pipe)

    管道的概述 管道也叫无名管道,它是是 UNIX 系统 IPC(进程间通信) 的最古老形式,全部的 UNIX 系统都支持这样的通信机制. 无名管道有例如以下特点: 1.半双工,数据在同一时刻仅仅能在一个 ...

  8. Linux系统编程——进程间通信:命名管道(FIFO)

    命名管道的概述 无名管道,因为没有名字,仅仅能用于亲缘关系的进程间通信(很多其它详情.请看<无名管道>).为了克服这个缺点.提出了命名管道(FIFO).也叫有名管道.FIFO 文件. 命名 ...

  9. Linux下进程间通信--共享内存:最快的进程间通信方式

    共享内存: 一.概念: 共享内存可以说是最有用的进程间通信方式,也是最快的IPC形式.两个不同进程A.B共享内存的意思是,同一块物理内存被映射到进程A.B各自的进程地址空间. 进程A可以即时看到进程B ...

随机推荐

  1. win10开机时内存使用率达到99%以上

    开始,运行,输入msconfig回车就能看到自启的项目. 搞定! 其实,感觉特别像是输入法的某个监听程序导致内存泄漏,造成的系统问题. 再遇到的时候要认真检查下.

  2. tomcat的连接数与线程池

    在使用tomcat时,经常会遇到连接数.线程数之类的配置问题,要真正理解这些概念,必须先了解Tomcat的连接器(Connector). 在前面的文章中写到过:Connector的主要功能,是接收连接 ...

  3. WinServer-IIS-woff字体不显示问题

    ASP.NET mvc发布到IIS之后,访问网站的时候,发现woff字体没有加载 百度发现很多博客上的教程是这样的,在IIS管理器中的MIME选项中添加类型 但是重新使用IIS发布后,新添加的字体就会 ...

  4. POJ 2154

    这题的时间卡的.... 必须用欧拉来优化,而且要加素数表.最重要是,因为最后结果要/n,而数据很大,所以,必须在之前就先/n了,否则会爆数据. #include <iostream> #i ...

  5. UIScrollView加入控件,控件距离顶部始终有间距的问题

    今天.特别郁闷.自己定义了一个UIScrollView,然后在它里面加入控件,如UIButton *button = [[UIButton alloc] initWithFrame:CGRectMak ...

  6. 闭包(closure)与协程共用时要注意的事情

    闭包是一种能够让你用非常舒服的方式来编程的小技巧,Go也支持闭包. 假设从来没有接触过闭包,想在一開始就弄懂什么是闭包(closure)是非常困难的,就像递归一样,直到你真正写过.用过它,你才干真正的 ...

  7. Android开发之使用Web Service进行网络编程

    使用Web Service进行网络编程 Android应用通常都是执行在手机平台上.手机系统的硬件资源是有限的,无论是存储能力还是计算能力都有限.在Android系统上开发.执行一些单用户.小型应用是 ...

  8. 修改YOLO使其显示自定义类别

    基本参考自这篇文章(http://blog.csdn.net/ma3252788/article/details/74659230),主要用来记录下自己遇到的问题 根据@赤战约风 的帖子做如下修改可以 ...

  9. 使用XMLHttpRequest解析json

    不适用内函数或者promise的方式,可以在外部提取到json数据 <!DOCTYPE html> <html lang="en"> <head> ...

  10. Linux Screen命令使程序远离断网影响

    linux中强大的screen命令2008-12-02 21:53今天发现了一个“宝贝”,就是Linux的screen命令,对于远程登录来说,不仅提供了类似于nohup的功能,而且提供了我非常喜欢的“ ...