共享内存就是允许两个或多个不相关的进程访问同一个逻辑内存。共享内存是在两个正在运行的进程之间共享和传递数据时,不需要在客户进程和服务器进程之间幅值,因此是最快的一种IPC。不同进程之间共享的内存通常安排为同一段物理内存。进程可以将同一段共享内存连接到它们自己的地址空间中,所有进程都可以访问共享内存中的地址,就好像它们是由用C语言函数malloc分配的内存一样。而如果某个进程向共享内存写入数据,所做的改动将立即影响到可以访问同一段共享内存的任何其他进程
 
注意:共享内存并未提供同步机制,也就是说,在第一个进程结束对共享内存的写操作之前,并无自动机制可以阻止第二个进程开始对它进行读取。所以我们通常需要用其他的机制来同步对共享内存的访问,例如前面说到的信号量。
 
接口函数
1、shmget函数
  1. #include <sys/ipc.h>
  2. #include <sys/shm.h>
  3.  
  4. int shmget(key_t key, size_t size, int shmflg);
  第一个参数,与信号量的semget函数一样,程序需要提供一个参数key(非0整数),它有效地为共享内存段命名,shmget函数成功时返回一个与key相关的共享内存标识符(非负整数),用于后续的共享内存函数。调用失败返回-1.
 
不相关的进程可以通过该函数的返回值访问同一共享内存,它代表程序可能要使用的某个资源,程序对所有共享内存的访问都是间接的,程序先通过调用shmget函数并提供一个键,再由系统生成一个相应的共享内存标识符(shmget函数的返回值),只有shmget函数才直接使用信号量键,所有其他的信号量函数使用由semget函数返回的信号量标识符。
 
第二个参数,size以字节为单位指定需要共享的内存容量
 
第三个参数,shmflg是权限标志,它的作用与open函数的mode参数一样,如果要想在key标识的共享内存不存在时,创建它的话,可以与IPC_CREAT做或操作。共享内存的权限标志与文件的读写权限一样,举例来说,0644,它表示允许一个进程创建的共享内存被内存创建者所拥有的进程向共享内存读取和写入数据,同时其他用户创建的进程只能读取共享内存。
 
2、shmat函数
  1. #include <sys/types.h>
  2. #include <sys/shm.h>
  3.  
  4. void *shmat(int shmid, const void *shmaddr, int shmflg);

  第一次创建完共享内存时,它还不能被任何进程访问,shmat函数的作用就是用来启动对该共享内存的访问,并把共享内存连接到当前进程的地址空间。

第一个参数,shm_id是由shmget函数返回的共享内存标识。
第二个参数,shm_addr指定共享内存连接到当前进程中的起始地址位置
      共享内存段链接到哪个地址与shmaddr参数以及在shmflag中是否指定SHM_RND位有关
      如果shmaddr为0,则此段连接到由内核选择的第一个可用地址上
      如果shmaddr非0,并且没有指定SHM_RND,则链接到shmaddr所指定的地址上
      如果shmaddr非0,且指定了SHM_RND,则链接的地址自动向下调整为SHMLBA的整数倍,公式为:(shmaddr-shmaddr% SHMLBA)
      通常shmaddr为0,表示让系统来选择共享内存的地址。
第三个参数,shm_flg是一组标志位,通常为0。

调用成功时返回一个指向共享内存第一个字节的指针,如果调用失败返回-1.

 
 3、shmdt函数

  1. #include <sys/types.h>
  2. #include <sys/shm.h>

  3. int shmdt(const void *shmaddr);

  该函数用于将共享内存从当前进程中分离。注意:将共享内存分离并不是删除它,只是使该共享内存对当前进程不再可用。

 参数shmaddr是shmat函数返回的地址指针
 调用成功时返回0,失败时返回-1.
 
4、shmctl函数
  1. #include <sys/ipc.h>
  2. #include <sys/shm.h>
  3.  
  4. int shmctl(int shmid, int cmd, struct shmid_ds *buf);

  与信号量的semctl函数一样,用来控制共享内存

第一个参数,shm_id是shmget函数返回的共享内存标识符。
 
第二个参数,command是要采取的操作,它可以取下面的三个值 :
    IPC_STAT:把shmid_ds结构中的数据设置为共享内存的当前关联值,即用共享内存的当前关联值覆盖shmid_ds的值。
    IPC_SET:如果进程有足够的权限,就把共享内存的当前关联值设置为shmid_ds结构中给出的值
    IPC_RMID:删除共享内存段
 
第三个参数,buf是一个结构指针,它指向共享内存模式和访问权限的结构。
shmid_ds结构至少包括以下成员:
  1. struct shmid_ds {
  2. struct ipc_perm shm_perm; /* Ownership and permissions */
  3. size_t shm_segsz; /* Size of segment (bytes) */
  4. time_t shm_atime; /* Last attach time */
  5. time_t shm_dtime; /* Last detach time */
  6. time_t shm_ctime; /* Last change time */
  7. pid_t shm_cpid; /* PID of creator */
  8. pid_t shm_lpid; /* PID of last shmat(2)/shmdt(2) */
  9. shmatt_t shm_nattch; /* No. of current attaches */
  10. ...
  11. };

  

示例

memory_write.c为创建共享内存并向里面写入由中断输入得数据,memory_read.c为读取共享内存中的内容。两个程序运行在两个不相关的进程中,为了保证两个进程间的读写同步,设置了WR_RD标记。

memory_write.c的程序如下

  1. #include <stdio.h>
  2. #include <sys/ipc.h>
  3. #include <sys/shm.h>
  4. #include <stdlib.h>
  5. #include <string.h>
  6.  
  7. #define MAX_SIZE 2048
  8. #define BUF_SIZE 2047
  9.  
  10. #define ERR_EXIT(m) \
  11. do \
  12. { \
  13. perror(m); \
  14. exit(EXIT_FAILURE); \
  15. }while(0)
  16.  
  17. struct share_memory
  18. {
  19. int WR_RD; //非0表示可写,0表示可读
  20. char text[MAX_SIZE];
  21. };
  22.  
  23. int main()
  24. {
  25. int shmid;
  26. char buffer[BUF_SIZE+1]; //用于保存输入的文本
  27. shmid = shmget(1234,sizeof(struct share_memory),0666 | IPC_CREAT); //创建共享内存
  28. if(shmid==-1)
  29. ERR_EXIT("shmget failed\n");
  30.  
  31. struct share_memory *sh_mem;
  32. void *sh=NULL;
  33. sh = shmat(shmid,0,0); //将共享内存连接到当前进程的地址空间
  34. if(sh ==(void *)-1)
  35. ERR_EXIT("shmat failed\n");
  36. else
  37. printf("Memory attached at9 %x\n",(int)sh);
  38.  
  39. sh_mem=(struct share_memory *)sh;
  40. sh_mem->WR_RD=1; // 置为可写
  41. int flag=1;
  42.  
  43. while(1)
  44. {
  45. while(!sh_mem->WR_RD) //共享内存为可读,表示内存里的数据还未被读取,不能写,此时堵塞
  46. {
  47. if(flag)
  48. {
  49. printf("memory is not empty,waiting.....\n");
  50. flag=0;
  51. }
  52.  
  53. sleep(1);
  54. }
  55.  
  56. flag=1;
  57. /*向共享内存写入数据*/
  58. printf("Enter some text(Enter 'end' to quit): \n");
  59. fgets(buffer,BUF_SIZE,stdin);
  60. strncpy(sh_mem->text,buffer,sizeof(buffer));
  61.  
  62. sh_mem->WR_RD=0; //写完数据之后设置共享内存可读
  63.  
  64. if(strncmp(buffer,"end",3)==0)
  65. break; //输入end结束循环
  66.  
  67. }
  68.  
  69. if(shmdt(sh)==-1)
  70. ERR_EXIT("shmdt failed\n"); //将共享内存从当前进程分离
  71.  
  72. sleep(2);
  73. return 0;
  74.  
  75. }

  

memory_read.c的程序如下

  1. #include <stdio.h>
  2. #include <sys/ipc.h>
  3. #include <sys/shm.h>
  4. #include <stdlib.h>
  5. #include <sys/types.h>
  6. #include <unistd.h>
  7.  
  8. #define MAX_SIZE 2048
  9.  
  10. #define ERR_EXIT(m) \
  11. do \
  12. { \
  13. perror(m); \
  14. exit(EXIT_FAILURE); \
  15. }while(0)
  16.  
  17. struct share_memory
  18. {
  19. int WR_RD; //非0表示可写,0表示可读
  20. char text[MAX_SIZE];
  21. };
  22.  
  23. int main()
  24. {
  25. int shmid;
  26.  
  27. srand(getpid());
  28.  
  29. shmid = shmget(1234,sizeof(struct share_memory),0666 | IPC_CREAT); //创建共享内存
  30. if(shmid==-1)
  31. ERR_EXIT("shmget failed\n");
  32.  
  33. struct share_memory *sh_mem;
  34. void *sh=NULL;
  35. sh = shmat(shmid,0,0); //将共享内存连接到当前进程的地址空间
  36. if(sh ==(void *)-1)
  37. ERR_EXIT("shmat failed\n");
  38. else
  39. printf("Memory attached at %x\n",(int)sh);
  40.  
  41. sh_mem=(struct share_memory*)sh;
  42. //sh_mem->WR_RD=ture; // 置为可写
  43.  
  44. while(1)
  45.  
  46. if(!sh_mem->WR_RD) //共享内存为可读,从共享内存读取数据
  47. {
  48. printf("your wrote: %s", sh_mem->text);
  49. sleep(rand()%3);
  50. sh_mem->WR_RD=1; //读完数据之后设置共享内存可写
  51.  
  52. if(strncmp(sh_mem->text,"end",3)==0)
  53. break; //输入end结束循环
  54. }
  55. else //表示内存为空,不能读,此时堵塞
  56. {
  57. //printf("memory is empty,waiting.....\n");
  58. sleep(1);
  59. }
  60.  
  61. if(shmdt(sh)==-1)
  62. ERR_EXIT("shmdt failed\n"); //将共享内存从当前进程分离
  63.  
  64. if(shmctl(shmid,IPC_RMID,0)==-1)
  65. ERR_EXIT("shmctl failed\n");
  66.  
  67. return 0;
  68.  
  69. }

  

使用共享内存的优缺点
 
1、优点:我们可以看到使用共享内存进行进程间的通信真的是非常方便,而且函数的接口也简单,数据的共享还使进程间的数据不用传送,而是直接访问内存,也加快了程序的效率。同时,它也不像匿名管道那样要求通信的进程有一定的父子关系。
 
2、缺点:共享内存没有提供同步的机制,这使得我们在使用共享内存进行进程间通信时,往往要借助其他的手段来进行进程间的同步工作。
 
 参考:
Linux进程间通信——使用共享内存       http://blog.csdn.net/ljianhui/article/details/10253345
 
Linux共享内存使用常见陷阱与分析      http://os.51cto.com/art/201311/418977_all.htm
 
 

IPC---共享内存的更多相关文章

  1. Linux IPC 共享内存

    共享内存 共享内存(shared memory)是最简单的Linux进程间通信方式之一. 使用共享内存,不同进程可以对同一块内存进行读写. 由于所有进程对共享内存的访问就和访问自己的内存空间一样,而不 ...

  2. linux IPC共享内存

    共享内存相关函数 获得一个共享存储标识符 #include <sys/ipc.h> #include <sys/shm.h int shmget(key_t key, size_t ...

  3. IPC——共享内存

    Linux进程间通信——使用共享内存 下面将讲解进程间通信的另一种方式,使用共享内存.   一.什么是共享内存 顾名思义,共享内存就是允许两个不相关的进程访问同一个逻辑内存.共享内存是在两个正在运行的 ...

  4. Linux环境进程间通信(五): 共享内存(下)

    linux下进程间通信的几种主要手段: 管道(Pipe)及有名管道(named pipe):管道可用于具有亲缘关系进程间的通信,有名管道克服了管道没有名字的限制,因此,除具有管道所具有的功能外,它还允 ...

  5. 【转载】ipcs与Linux共享内存

    一.共享内存相关知识 所谓共享内存,就是多个进程间共同地使用同一段物理内存空间,它是通过将同一段物理内存映射到不同进程的 虚拟空间来实现的.由于映射到不同进程的虚拟空间中,不同进程可以直接使用,不需要 ...

  6. 进程间通信之信号量、消息队列、共享内存(system v的shm和mmap)+信号signal

    进程间通信方式有:System v unix提供3种进程间通信IPC:信号量.消息队列.共享内存.此外,传统方法:信号.管道.socket套接字. [注意上述6种方式只能用户层进程间通信.内核内部有类 ...

  7. <转>Linux环境进程间通信(五): 共享内存(下)

    http://www.ibm.com/developerworks/cn/linux/l-ipc/part5/index2.html 系统调用mmap()通过映射一个普通文件实现共享内存.系统V则是通 ...

  8. Linux环境进程间通信: 共享内存

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

  9. (转)Linux环境进程间通信系列(五):共享内存

    原文地址:http://www.cppblog.com/mydriverc/articles/29741.html 共享内存可以说是最有用的进程间通信方式,也是最快的 IPC 形式.两个不同进程 A ...

  10. Linux进程间通信(四) - 共享内存

    共享内存的优势 采用共享内存通信的一个显而易见的好处是效率高,因为进程可以直接读写内存,而不需要任何数据的拷贝.对于像管道和消息队列等通信方式,则需要在内核和用户空间进行四次的数据拷贝,而共享内存则只 ...

随机推荐

  1. Autofac和DynamicProxy2搭配实现Aop动手训练

    http://www.cnblogs.com/zhengwl/p/5433181.html Aop含义:aspect-oriented programming 实现工具介绍 Autofac是一个比较流 ...

  2. 使用Xunit来进行单元测试

    不管你爱与不爱,单元测试对于一个软件的长治久安还是必不可少的一环.在Visual Studio 2012后,VS中的测试浏览器也能与第三方的集成了,用起来还是非常方便的.目前在.Net框架下的测试工具 ...

  3. InstallShield 2010 使用 .net framework 4.5

    一.InstallShield 2010 使用 .net framework 4.5记录 1.prq的地址,通过以下地址,下载相应的prq文件 .NET 4.5: http://saturn.inst ...

  4. Spring利用JDBCTemplate实现批量插入和返回id

    1.先介绍一下java.sql.Connection接口提供的三个在执行插入语句后可取的自动生成的主键的方法: //第一个是 PreparedStatement prepareStatement(St ...

  5. [Html5]sessionStorage和localStorage的区别

    摘要 有时需要在浏览器中保存一些数据,特别在app中嵌入的h5页面中,需要在webview中保存一些数据,作为客户端的数据持久化. h5中web storage有两种存储方式:sessionStora ...

  6. Java Programming Test Question 1

    public class JPTQuestion1 { public static void main(String[] args) { String s1 = "abc"; St ...

  7. winform 开发心得~

    winform自适应不同分辨率 不同dpi 1.窗体AutoScaleMode属性 使用None 2.自定义控件 AutoScaleMode 使用Inherit 3.所有控件窗体字体使用px为单位

  8. gitlab一键安装 (转)

    原文地址:http://www.2cto.com/os/201411/353292.html 0 简介bitnami和gitlab bitnami BitNami是一个开源项目,该项目产生的开源软件包 ...

  9. word文档的生成、修改、渲染、打印,使用Aspose.Words

    无需MS Word也可执行各种文档处理任务,包括文档的生成.修改.渲染.打印,文档格式转换和邮件合并等文档处理.

  10. Codeforces 260 C. Boredom

    题目链接:http://codeforces.com/contest/456/problem/C 解题报告:给出一个序列,然后选择其中的一个数 k 删除,删除的同时要把k - 1和k + 1也删除掉, ...