共享内存是被多个进程共享的一部分物理内存。共享内存是进程间共享数据的一种最快的方法。一个进程向共享内存区域写入了数据,共享这个内存区域的全部进程就能够立马看到当中的内容。



关于共享内存使用的API

key_t ftok(const char *pathname, int proj_id);

#在IPC中。我们经经常使用一个 key_t 的值来创建或者打开 信号量。共享内存和消息队列。这个 key_t 就是由ftok函数产生的。

pathname:指定的文件名称,该文件必须是存在并且能够訪问

proj_id:1~255之间的整数值。

对于ftok这个函数,个人是认为没有太大用处。第一,应用程序可能会在不同的主机上使用,(换了一个环境。文件有没有?)。第二点,假设在訪问同一共享内存的多个进程先后调用ftok这个时间段中, pathname指定的文件被删除且又一次创建,那么,每一个进程得到的 key_t 是不一样的。应用程序不会报错。可是数据共享的目的是达不到了。









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

shmget 用来获得共享内存区域的ID。假设不存在指定的共享区域就创建对应的区域。

key: 这块共享内存的标识符。

假设是父子关键的进程间通信,这个标识符用 IPC_PRIVATE 取代。

假设两个进程没有不论什么关系,官方的说法是用ftok产生一个key(鉴于上面关于ftok的介绍。个人比較认同的做法是自定义一个)使用。

size:这块共享内存的大小。

(字节数)

shmflg:是一组标志。

IPC_CREAT:假设共享内存不存在。则创建一个共享内存。

IPC_EXCL:仅仅有在共享内存不存在的时候。新的共享内存才建立。否则就产生错误。

对于这个參数通常是这样操作

#define PERM S_IRUSR | S_IWUSR | IPC_CREAT

然后把 PERM 当作shmflg。

成功返回共享内存的标识符。不成功返回-1,并设置errno。









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

用来建立内存映射(同意本进程与指定共享内存建立联系。读写数据)

shmid:共享内存的ID。(shmget函数的返回值)

shmaddr:指定共享内存连接到当前进程中的地址位置,通常为空,表示让系统选择共享内存的地址。

shmflg:是一组标致。代表本进程对该内存进程的操作模式。假设是SHM_RDONLY的话就是仅仅读模式,其它的是读写模式。

这个參数一般设置为0。

成功返回共享内存的起始地址,失败返回-1,并设置errno。





int shmdt(const void *shmaddr);

断开链接的共享内存指针。(并非从内核正真的删除这个共享内存段)

shmaddr:共享内存的起始地址(shmat函数的返回值)





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

控制共享内存的使用

shmid:共享内存的ID。(shmget函数的返回值)

cmd: 控制命令。可取值例如以下:

IPC_STAT:得到共享内存的状态

IPC_SET:改变共享内存的状态

IPC_RMID:删除共享内存

buf:是一个结构体指针,当cmd为IPC_STAT的时候,取得的状态放入这个结构体中。

假设要改变共享内存的状态,用这个结构体指定。

struct shmid_ds 原型

struct shmid_ds {

    struct ipc_perm shm_perm;    /* 操作权限 */

    size_t          shm_segsz;   /* 共享内存段的大小(以字节为单位) */

    time_t          shm_atime;   /* 最后一个进程附加到该段的时间 */

    time_t          shm_dtime;   /* 最后一个进程离开该段的时间 */

    time_t          shm_ctime;   /* 最后一个进程改动该段的时间 */

    pid_t           shm_cpid;    /* 创建共享内存段进程的pid */

    pid_t           shm_lpid;    /* 在该共享内存段上操作的最后一个进程的pid */

    shmatt_t        shm_nattch;  /* 当前附件到该共享内存段的进程个数 */

    ...

};

成功返回0。失败返回-1。并设置errno。

注意!

!!!

!!

!!:在使用共享内存,结束程序退出后。假设你没在程序中用shmctl()删除共享内存的话,一定要在命令行下用ipcrm命令删除这块共享内存。你要是无论的话,它就一直在那儿放着了。

简单解释一下ipcs命令和ipcrm命令。





应用场景

  • 进程间通讯-生产者消费者模式
生产者进程和消费者进程通信常使用共享内存,比方一个网络server。接入进程收到的数据包后。直接写到共享内存中。并唤醒处理进程,处理进程从共享内存中读数据包。进行处理。当然,这里要解决相互排斥的问题。

演示样例程序:
  1. #ifndef SHM_COM_H_INCLUDED
  2. #因为fork产生的子进程和父进程不共享内存区,所以父子进程间的通讯也能够使用共享内存。
  3.  
  4. 父进程启动后创建内存共享,然后调用fork创建子进程。最后分别在父子进程做内存映射。以达到数据共享。
  5.  
  6. define SHM_COM_H_INCLUDED
  7.  
  8. #include <stdio.h>
  9. #include <unistd.h>
  10. #include <string.h>
  11. #include <stdlib.h>
  12. #include <errno.h>
  13. #include <sys/ipc.h>
  14. #include <sys/shm.h>
  15. #include <sys/types.h>
  16. #include <sys/stat.h>
  17.  
  18. #define PERM S_IRUSR | S_IWUSR | IPC_CREAT
  19.  
  20. #define TEXT_SZ 2048
  21.  
  22. struct shared_use_st
  23. {
  24. int written_by_you;
  25. char some_text[TEXT_SZ];
  26. };


  1. /*
  2. 消费者程序
  3.  
  4. 创建共享内存段
  5. 然后将它连接到它自己的地址空间中(共享内存映射)
  6. 而且。
  7. 我们在共享内存的開始处使用了一个结构shared_use_st.
  8. 该结构中有个标志written_by_you,
  9. 当共享内存中有数据写入时,就设置这个标志。
  10.  
  11. 这个标志被设置时,
  12. 程序就从共享内存中读取文本,
  13. 将它打印出来,
  14. 然后清除这个标志。表示已经读完数据。
  15. 我们用一个特殊字符串end来退出循环。
  16.  
  17. 接下来,
  18. 程序分离共享内存段并删除它。
  19. */
  20.  
  21. #include "shm_com.h"
  22.  
  23. int main ( int argc, char** argv )
  24. {
  25. srand ( ( unsigned int ) getpid() );
  26. int runing = 1, shmid;
  27. struct shared_use_st *shared_stuff;
  28.  
  29. if ( ( shmid = shmget ( ( key_t ) 8888, sizeof ( struct shared_use_st ), PERM ) ) == -1 )
  30. {
  31. fprintf ( stderr, "%s %s %d\n", strerror ( errno ), __FILE__, __LINE__ );
  32. return -1;
  33. }
  34.  
  35. if ( ( shared_stuff = ( struct shared_use_st * ) shmat ( shmid, 0, 0 ) ) == ( struct shared_use_st * ) - 1 )
  36. {
  37. fprintf ( stderr, "%s %s %d\n", strerror ( errno ), __FILE__, __LINE__ );
  38. return -1;
  39. }
  40.  
  41. shared_stuff->written_by_you = 0;
  42.  
  43. while ( runing )
  44. {
  45. while(shared_stuff->written_by_you == 0)
  46. sleep(1);
  47.  
  48. printf ( "You wrote: %s", shared_stuff->some_text );
  49. sleep ( rand() % 4 );
  50. shared_stuff->written_by_you = 0;
  51.  
  52. if ( strncmp ( shared_stuff->some_text, "end", 3 ) == 0 )
  53. {
  54. runing = 0;
  55. }
  56. }
  57.  
  58. if ( shmdt ( shared_stuff ) == -1 )
  59. {
  60. fprintf ( stderr, "%s %s %d\n", strerror ( errno ), __FILE__, __LINE__ );
  61. return -1;进程间共享-仅仅读模式业务常常碰到一种场景,进程须要载入一份配置文件。可能这个文件有100K大,那假设这台机器上多个进程多要载入这份配置文件。比方有200个进程,那么内存开销合计为20M,但假设文件很多其它或者进程数很多其它时,这样的对内存的消耗就是一种严重的浪费。比較好的解决方法是,由一个进程负责把配置文件载入到共享内存中,然后全部须要这份配置的进程仅仅要使用这个共享内存就可以。
  62. }
  63.  
  64. if ( shmctl ( shmid, IPC_RMID, 0 ) == -1 )
  65. {
  66. fprintf ( stderr, "%s %s %d\n", strerror ( errno ), __FILE__, __LINE__ );
  67. return -1;
  68. }
  69.  
  70. return 0;
  71. }
  1. #include "shm_com.h"
  2.  
  3. /*
  4. 生产者程序
  5. 使用同样的键值来取得并连接同一个共享内存段,
  6. 然后提示用户输入一些文本。
  7. 假设标志written_by_you被设置,
  8. 生产者就知道消费都进程还未读完上一次的数据,
  9. 因此就继续等待。
  10. 当其他进程清除了这个标志后,
  11. 生产者写入新的数据并设置这个标志。
  12. 它还使用字符串end来终止并分离共享内存段。
  13.  
  14. 这里提供的同步标志written_by_you。
  15. 它是一个很缺乏效率的忙等待(不停地循环)。
  16.  
  17. */
  18. int main ( int argc, char** argv )
  19. {
  20. int running = 1, shmid;
  21. struct shared_use_st *shared_stuff;
  22.  
  23. if ( ( shmid = shmget ( ( key_t ) 8888, sizeof ( struct shared_use_st ), PERM ) ) == -1 )
  24. {
  25. fprintf ( stderr, "%s %s %d\n", strerror ( errno ), __FILE__, __LINE__ );
  26. return -1;
  27. }
  28.  
  29. if ( ( shared_stuff = ( struct shared_use_st * ) shmat ( shmid, 0, 0 ) ) == ( struct shared_use_st * ) - 1 )
  30. {
  31. fprintf ( stderr, "%s %s %d\n", strerror ( errno ), __FILE__, __LINE__ );
  32. return -1;
  33. }
  34.  
  35. while ( running )
  36. {
  37. while ( shared_stuff->written_by_you == 1 )
  38. {
  39. sleep ( 1 );
  40. }
  41.  
  42. printf ( "Enter sone text: " );
  43. memset ( shared_stuff->some_text, 0, sizeof ( shared_stuff->some_text ) );
  44. fgets ( shared_stuff->some_text, TEXT_SZ, stdin );
  45. shared_stuff->written_by_you = 1;
  46.  
  47. if ( strncmp ( shared_stuff->some_text, "end", 3 ) == 0 )
  48. {
  49. running = 0;
  50. }
  51. }
  52.  
  53. if ( shmdt ( shared_stuff ) == -1 )
  54. {
  55. fprintf ( stderr, "%s %s %d\n", strerror ( errno ), __FILE__, __LINE__ );
  56. return -1;
  57. }
  58.  
  59. return 0;
  60. }
  • 父子进程间通讯
因为fork产生的子进程和父进程不共享内存区,所以父子进程间的通讯也能够使用共享内存。父进程启动后创建内存共享,然后调用fork创建子进程。

最后分别在父子进程做内存映射。以达到数据共享。

演示样例代码:
  1. #include <unistd.h>
  2. #include <sys/stat.h>
  3. #include <sys/types.h>
  4. #include <sys/ipc.h>
  5. #include <sys/shm.h>
  6. #include <stdio.h>
  7. #include <string.h>
  8. #include <errno.h>
  9. #include <stdlib.h>
  10.  
  11. #define PERM S_IRUSR | S_IWUSR | IPC_CREAT
  12. typedef char * p_str;
  13. int main ( int argc, char **argv )
  14. {
  15. int shmid;
  16. pid_t pid;
  17. p_str p_shmaddr, c_shmaddr;
  18.  
  19. if ( ( shmid = shmget ( IPC_PRIVATE, 1024, PERM ) ) == -1 )
  20. {
  21. fprintf ( stderr, "%s %s %d\n", strerror ( errno ), __FILE__, __LINE__ );
  22. return -1;
  23. }
  24.  
  25. if ( ( pid = fork() ) == 0 )
  26. {
  27. if ( ( c_shmaddr = shmat ( shmid, 0, 0 ) ) == ( p_str ) ( -1 ) )
  28. {
  29. fprintf ( stderr, "%s %s %d\n", strerror ( errno ), __FILE__, __LINE__ );
  30. return -1;
  31. }
  32.  
  33. memset ( c_shmaddr, '\0', 1024 );
  34. stpcpy ( c_shmaddr, "Hello" );
  35.  
  36. if ( shmdt ( c_shmaddr ) == -1 )
  37. {
  38. fprintf ( stderr, "%s %s %d\n", strerror ( errno ), __FILE__, __LINE__ );
  39. return -1;
  40. }
  41.  
  42. wait ( NULL );
  43. }
  44. else
  45. if ( pid > 0 )
  46. {
  47. sleep ( 2 );
  48.  
  49. if ( ( p_shmaddr = shmat ( shmid, 0, 0 ) ) == ( p_str ) ( -1 ) )
  50. {
  51. fprintf ( stderr, "%s %s %d\n", strerror ( errno ), __FILE__, __LINE__ );
  52. return -1;
  53. }
  54.  
  55. printf ( "%s\n", p_shmaddr );
  56.  
  57. if ( shmdt ( p_shmaddr ) == -1 )
  58. {
  59. fprintf ( stderr, "%s %s %d\n", strerror ( errno ), __FILE__, __LINE__ );
  60. return -1;
  61. }
  62.  
  63. if ( shmctl ( shmid, IPC_RMID, 0 ) == -1 )
  64. {
  65. fprintf ( stderr, "%s %s %d\n", strerror ( errno ), __FILE__, __LINE__ );
  66. return -1;
  67. }
  68.  
  69. exit ( 0 );
  70. }
  71. else
  72. {
  73. fprintf ( stderr, "%s %s %d\n", strerror ( errno ), __FILE__, __LINE__ );
  74. return -1;
  75. }
  76.  
  77. return 0;
  78. }

  • 进程间共享-仅仅读模式
业务常常碰到一种场景。进程须要载入一份配置文件,可能这个文件有100K大。那假设这台机器上多个进程多要载入这份配置文件,比方有200个进程,那么内存开销合计为20M,但假设文件很多其它或者进程数很多其它时,这样的对内存的消耗就是一种严重的浪费。

比較好的解决方法是,由一个进程负责把配置文件载入到共享内存中。然后全部须要这份配置的进程仅仅要使用这个共享内存就可以。


代码临时不想写了。。

。有须要的时候再补上吧。。

linux 进程通信之 共享内存的更多相关文章

  1. Linux 进程通信(共享内存区)

    共享内存是由内核出于在多个进程间交换信息的目的而留出的一块内存区(段). 如果段的权限设置恰当,每个要访问该段内存的进程都可以把它映像到自己的私有地址空间中. 如果一个进程更新了段中的数据,其他进程也 ...

  2. linux进程通信之共享内存

    共享内存同意两个或多个进程共享一给定的存储区,由于数据不须要来回复制,所以是最快的一种进程间通信机制.共享内存能够通过mmap()映射普通文件(特殊情况下还能够採用匿名映射)机制实现,也能够通过系统V ...

  3. Linux进程通信之共享内存实现生产者/消费者模式

    共享内存 共享内存是内核为进程创建的一个特殊内存段,它将出现在进程自己的地址空间中,其它进程可以将同一段共享内存连接(attach)到自己的地址空间.这是最快的进程间通信方式,但是不提供任何同步功能( ...

  4. linux 进程学习笔记-共享内存

    如果能划定一块物理内存,让多个进程都能将该内存映射到其自身虚拟内存空间的话,那么进程可以通过向这块内存空间读写数据而达到通信的目的.另外,和消息队列不同的是,共享的内存在用户空间而不是核空间,那么就不 ...

  5. Linux 进程通信之:内存共享(Shared Memory)(转,好文章)

    https://blog.csdn.net/afei__/article/details/84188548

  6. Linux进程通信之System V共享内存

    前面已经介绍过了POSIX共享内存区,System V共享内存区在概念上类似POSIX共享内存区,POSIX共享内存区的使用是调用shm_open创建共享内存区后调用mmap进行内存区的映射,而Sys ...

  7. 撸代码--linux进程通信(基于共享内存)

    1.实现亲缘关系进程的通信,父写子读 思路分析:1)首先我们须要创建一个共享内存. 2)父子进程的创建要用到fork函数.fork函数创建后,两个进程分别独立的执行. 3)父进程完毕写的内容.同一时候 ...

  8. linux进程间的通信之 共享内存

    一.共享内存介绍 共享内存是三个IPC(Inter-Process Communication)机制中的一个. 它允许两个不相关的进程访问同一个逻辑内存. 共享内存是在两个正在进行的进程之间传递数据的 ...

  9. 【网络编程基础】Linux下进程通信方式(共享内存,管道,消息队列,Socket)

    在网络课程中,有讲到Socket编程,对于tcp讲解的环节,为了加深理解,自己写了Linux下进程Socket通信,在学习的过程中,又接触到了其它的几种方式.记录一下. 管道通信(匿名,有名) 管道通 ...

随机推荐

  1. POJ 3855 计算几何·多边形重心

    思路: 多边形面积->任选一个点,把多边形拆成三角,叉积一下 三角形重心->(x1+x2+x3)/3,(y1+y2+y3)/3 多边形重心公式题目中有,套一下就好了 计算多边形重心方法: ...

  2. VMware 14 Pro 永久许可证激活密钥

    VMware 14 Pro 永久许可证激活密钥 FF31K-AHZD1-H8ETZ-8WWEZ-WUUVACV7T2-6WY5Q-48EWP-ZXY7X-QGUWD

  3. 话说:Hibernate二级缓存

    Hibernate缓存分类: 一.Session缓存(又称作事务缓存):Hibernate内置的,不能卸除. 缓存范围:缓存只能被当前Session对象访问.缓存的生命周期依赖于Session的生命周 ...

  4. jQuery五屏轮播手风琴切换代码

    jQuery五屏轮播手风琴切换代码 在线演示本地下载

  5. 关于idea failed to start SceneBuilder 的解决方法

    问题描述: javaFx无法启动SceneBuilder. 问题原因: SceneBuilder不正当配置. 解决方法:1. 下载SceneBuilder   https://www.oracle.c ...

  6. activiti 表——介绍

    Activiti 用到的表都是act_开头 下面分别来介绍下 activiti 所用到的表: Act_RE_* 表示 RepositoryService接口所操作的表:包含了流程定义信息 .流程静态资 ...

  7. Android Activity作为dialog对话框的使用详细介绍

    Activity做为Android应用层四大组件的重要组成部分,它的灵活性.可扩性无论是在应用还是游戏方面都得到了广泛应用,本文主要介绍Activity作为dialog对话框 的使用方法进行说明. 本 ...

  8. 编写第一个HTML5文件

    1.3.1  HTML文件的编写方法 编写HTML文件主要有如下3种方法: 手工直接编写 由于HTML语言编写的文件是标准的ASCII文本文件,所以我们可以使用任何的文本编辑器来打开并编写HTML文件 ...

  9. jQuery与js的区别,并有基本语法详解,

    通过过一下对比,我们能很清楚的发现jquery与js的区别,运用jquery能大量减少代码量,不过js里面关于时间的setinterval和settimeout只能用js <script src ...

  10. Sql Server 优化 SQL 查询:如何写出高性能SQL语句

    1. 首先要搞明白什么叫执行计划? 执行计划是数据库根据SQL语句和相关表的统计信息作出的一个查询方案,这个方案是由查询优化器自动分析产生的,比如一条SQL语句如果用来从一个 10万条记录的表中查1条 ...