一,共享内存介绍
共享内存是三个IPC(Inter-Process Communication)机制中的一个,它允许两个不相关的进程访问同一个逻辑内存。
 

二、共享内存使用的函数

#include <sys/shm.h>
  int shmget(key_t key, size_t size, int shmflg);
  void *shmat(int shm_id, const void *shm_addr, int shmflg);
  int shmdt(const void *shm_addr);
  int shmctl(int shm_id, int cmd, struct shmid_ds *buf);
 

1. shmget函数
该函数用来创建共享内存:
      int shmget(key_t key, size_t size, int shmflg);
参数:
  key : 和信号量一样,程序需要提供一个参数key,它有效地为共享内存段命名。有一个特殊的键值IPC_PRIVATE, 它用于创建一个只属于创建进程的共享内存,通常不会用到。
  size: 以字节为单位指定需要共享的内存容量。
  shmflag: 包含9个比特的权限标志,它们的作用与创建文件时使用的mode标志是一样。由IPC_CREAT定义的一个特殊比特必须和权限标志按位或才能创建一个新的共享内存段。

NOTE:
  权限标志对共享内存非常有用,因为它允许一个进程创建的共享内存可以被共享内存的创建者所拥有的进程写入,同时其它用户创建的进程只能读取共享内存。
  我们可以利用这个功能来提供一种有效的对数据进行只读访问的方法,通过将数据放共享内存并设置它的权限,就可以避免数据被其他用户修改。

返回值:
创建成功,则返回一个非负整数,即共享内存标识;
如果失败,则返回-1.

2. shmat函数
第一次创建共享内存段时,它不能被任何进程访问。要想启动对该内存的访问,必须将其连接到一个进程的地址空间。
这个工作由shmat函数完成:
    void *shmat(int shm_id, const void *shm_addr, int shmflg);
参数:
  shm_id : 由shmget返回的共享内存标识。
  shm_add: 指定共享内存连接到当前进程中的地址位置。它通常是一个空指针, 表示让系统来选择共享内存出现的地址。
  shmflg : 是一组标志。它的两个可能取值是:SHM_RND, 和shm_add联合使用,用来控制共享内存连接的地址。SHM_RDONLY, 它使连接的内存只读
         
返回值:
如果调用成功, 返回一个指向共享内存第一个字节的指针;
如果失败,返回-1.

NOTE:
  共享内存的读写权限由它的属主(共享内存的创建者),
  它的访问权限和当前进程的属主决定。
  共享内存的访问权限类似于文件的访问权限。

3. shmdt函数
将共享内存从当前进程中分离。
    int shmdt(const void *shm_addr);
  shm_addr: shmat返回的地址指针。
成功时,返回0,
失败时,返回-1.

NOTE:
  共享内存分离并未删除它,
  只是使得该共享内存对当前进程不再可用。

4. shmctl函数
共享内存的控制函数
    int shmctl(int shm_id, int cmd, struct shmid_ds *buf);
   shmid_ds结构至少包含以下成员:
    struct shmid_ds {
      uid_t shm_perm.uid;
      uid_t shm_perm.gid;
      mode_t shm_perm.mode;
    }
参数:
  shm_id : 是shmget返回的共享内存标识符。
  command: 是要采取的动作,
         它可以取3个值:
    IPC_STAT  把shmid_ds结构中的数据设置为共享内存的当前关联值
    IPC_SET   如果进程有足够的权限, 就把共享内存的当前关联值设置为shmid_ds结构中给出的值
    IPC_RMID  删除共享内存段
  buf  : 是一个指针,包含共享内存模式和访问权限的结构。

返回值:
成功时,返回0,
失败时,返回-1

例:生产者和消费者实例

头文件:shm_com.h

shm1.c 生产者程序通过它向消费者程序输入数据。

shm2.c 消费者将创建一个共享内存段,然后把写到它里面的数据都显示出来。

 #define TEXT_SZ 2048
 struct shared_use_st {
      int written;
      char some_text[TEXT_SZ];
  };

shm_com.h

 //shm1.c 生产者程序通过它向消费者程序输入数据。
 #include <unistd.h>
 #include <stdlib.h>
 #include <stdio.h>
 #include <string.h>
 #include <sys/shm.h>
 #include "shm_com.h"
 int main()
 {
       ;
       ;
       struct shared_use_st *shared_stuff;
       char buffer[BUFSIZ];
       int shmid;

       shmid = shmget((key_t),  | IPC_CREAT);
       )
       {
         fprintf(stderr, "shmget failed\n");
         exit(EXIT_FAILURE);
       }

       shared_memory = shmat(shmid, (, );
       )
       {
         fprintf(stderr, "shmat failed\n");
         exit(EXIT_FAILURE);
       }

       printf("Memory attached at %X\n", (int)shared_memory);

       shared_stuff = (struct shared_use_st *)shared_memory;
       while(running)
       {
         )
         {
           sleep();
           printf("waiting for client...\n");
         }
         printf("Enter some text: ");
         fgets(buffer, BUFSIZ, stdin);

         strncpy(shared_stuff->some_text, buffer, TEXT_SZ);
         shared_stuff->written = ;

         ) == ) {
           running = ;
         }
       }

       ) {
         fprintf(stderr, "shmdt failed\n");
         exit(EXIT_FAILURE);
       }

       exit(EXIT_SUCCESS);
     }

shm1.c

 //shm2.c 消费者将创建一个共享内存段,然后把写到它里面的数据都显示出来。
 #include <unistd.h>
 #include <stdlib.h>
 #include <stdio.h>
 #include <string.h>
 #include <sys/shm.h>
 #include "shm_com.h"
 int main()
 {
     ;
     ;
     struct shared_use_st *shared_stuff;
     int shmid;
     srand((unsigned int)getpid());
     shmid = shmget((key_t),  | IPC_CREAT);

     ) {
     fprintf(stderr, "shmget failed\n");
     exit(EXIT_FAILURE);
     }

     //现在,让程序可以访问这个共享内存:
     shared_memory = shmat(shmid, (, );

     ) {
     fprintf(stderr, "shmat failed\n");
     exit(EXIT_FAILURE);
     }
     printf("Memory attached at %X\n", (int)shared_memory);

     //程序的下一部分将shared_memory分配给shared_stuff,然后它输出written中的文本。
     //循环将一直执行到在written中找到end字符串为止。
     //sleep调用强迫消费者程序在临界区域多待一会,
     //让生产者程序等待:

     shared_stuff = (struct shared_use_st *)shared_memory;
     shared_stuff->written = ;

     while(running)
     {
     if (shared_stuff->written)
     {
       printf("You wrote: %s", shared_stuff->some_text);

       sleep( rand() %  ); /* make the other process wait for us ! */
       shared_stuff->written = ;

       ) == ) {
         running = ;
       }
     }
     }

     //最后,共享内存被分离,然后被删除:
     )
     {
     fprintf(stderr, "shmdt failed\n");
     exit(EXIT_FAILURE);
     }

     ) == -)
     {
     fprintf(stderr, "shmctl(IPC_RMID) failed\n");
     exit(EXIT_FAILURE);
     }

     exit(EXIT_SUCCESS);
 }

shm2.c

程序解析:
消费者程序
1)创建共享内存段,
2)然后将它连接到它自己的地址空间中,
3)我们在共享内存的开始处使用了一个结构shared_use_st.该结构中有个标志written,当共享内存中有数据写入时,就设置这个标志。
这个标志被设置时,程序就从共享内存中读取文本,将它打印出来,然后清除这个标志,表示已经读完数据。我们用一个特殊字符串end来退出循环。
4)程序分离共享内存段并删除它。

生产者程序
1)使用相同的键值1234来取得并连接同一个共享内存段,
2)然后提示用户输入一些文本。如果标志written被设置,生产者就知道消费都进程还未读完上一次的数据,因此就继续等待。
3)当其它进程清除了这个标志后,生产者写入新的数据并设置这个标志。它还使用字符串end来终止并分离共享内存段。

这里提供的同步标志written,它是一个非常缺乏效率的忙等待(不停地循环)

IPC编程之共享内存的更多相关文章

  1. linux编程之共享内存

    linux 进程间通信(IPC)包括3种机制:消息队列.信号量.共享内存.消息队列和信号量均是内核空间的系统对象,经由它们 的数据需要在内核和用户空间进行额外的数据拷贝:而共享内存和访问它的所有应用程 ...

  2. Linux IPC System V 共享内存

    模型 #include<sys/types.h> #include<sys/ipc.h> #include<sys/shm.h> ftok() //获取key值 s ...

  3. Linux IPC实践(8) --共享内存/内存映射

    概述 共享内存区是最快的IPC形式.一旦这样的内存映射到共享它的进程的地址空间,这些进程间数据传递不再涉及到内核,换句话说是进程不再通过执行进入内核的系统调用来传递彼此的数据(如图). 共享内存 VS ...

  4. Unix IPC之基于共享内存的计数器

    目的 本文主要实现一个基于共享内存的计数器,通过父子进程对其访问. 本文程序需基于<<Unix网络编程-卷2>>的环境才能运行.程序中大写开头的函数为其小写同名函数的包裹函数, ...

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

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

  6. IPC 进程间通信方式——共享内存

    共享内存 共享内存区域是被多个进程共享的一部分物理内存. 多个进程都可以把共享内存映射到自己的虚拟空间.所有用户空间的进程要操作共享内存,都要将其映射到自己的虚拟空间,通过映射的虚拟内存空间地址去操作 ...

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

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

  8. System V IPC 之共享内存

    IPC 是进程间通信(Interprocess Communication)的缩写,通常指允许用户态进程执行系列操作的一组机制: 通过信号量与其他进程进行同步 向其他进程发送消息或者从其他进程接收消息 ...

  9. C 共享内存封装

    引言 - 背景 2016 年写过一篇关于 linux 共享内存 shm api 扫盲文. C扩展 从共享内存shm到memcache外部内存 比较简单. 没有深入分析(能力有限, 也深入分析不了). ...

随机推荐

  1. 深度问答之提取语料,导入了yml模块

    根据目录下的yml文件循环创建同名文件夹,并从yml文件读取问答并给每个文件夹写入question和answer文件 #!/usr/bin/env python # coding:utf8 # aut ...

  2. 牛客练习赛13 B 幸运数字Ⅱ 【暴力】【二分】

    题目链接 https://www.nowcoder.com/acm/contest/70/B 思路 没有代码限制 先打表 打出 幸运数字的表 然后 二分查找 第一个 大于 r 的幸运数字 然后 往 L ...

  3. vim终端配色(非gui版本)——Monokai

    啥也别说,先上图. 具体配置: 1. 将molokai.vim文件(下面贴出)放到 ~/.vim/colors 目录下,如没有此文件夹需自行创建. 提示:~ 代表用户主目录,如我的用户名是 akaed ...

  4. [原创]Scala学习:流程控制,异常处理

    1.流程控制 1)do..while def doWhile(){ var line="" do{ line = readLine() println("readline ...

  5. Python 3 udp 套接字

    Python 3 udp套接字 TCP是建立可靠连接,并且通信双方都可以以流的形式发送数据.相对TCP,UDP则是面向无连接的协议 使用UDP协议时,不需要建立连接,只需要知道对方的IP地址和端口号, ...

  6. day3--集合、文件操作、字符编码与转换、函数(递归,lambda,filter,map)、字典排序

    list1 = set([1, 2, 3, 4, 5, 6, 5, 5, 5])list2 = set([11, 2, 36, 'a', 5, 6, 5, 5, 5])list3 = set([1, ...

  7. Quartz.Net在C#中的使用

    概述 Quartz.NET是一个开源的作业调度框架,非常适合在平时的工作中,定时轮询数据库同步,定时邮件通知,定时处理数据等. Quartz.NET允许开发人员根据时间间隔(或天)来调度作业.它实现了 ...

  8. sqlserver 2008 创建数据库的时候不是空库,里面总有数据的解决办法

    SqlServer2008 里面有个系统数据库 Model 数据库,在创建新数据库的时候,会以它为模板创建,所以如果发现你的Model数据库比较大,说明里面有很多模板数据.此时如果需要去创建没有数据的 ...

  9. logstash的output插件

    logstash 的output插件 nginx,logstash和redis在同一台机子上 yum -y install redis,vim /etc/redis.conf 设置bind 0.0.0 ...

  10. HDU 4001 To Miss Our Children Time(2011年大连网络赛 A 贪心+dp)

    开始还觉得是贪心呢... 给你三类积木叫你叠楼房,给你的每个积木包括四个值:长 宽(可以互换) 高 类型d d=0:你只能把它放在地上或者放在 长 宽 小于等于 自己的积木上面 d=1:你只能把它放在 ...