共享内存(共享内存的映射,用共享内存读写,共享内存的控制)

一、共享内存概述

共享内存允许两个或者多个进程共享给定的存储区域。

共享内存的特点

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

一个进程向共享的内存区域写入了数据,共享这个内存区域的所有进程就可以立刻看到 其中的内容。

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

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

共享内存示意图

 总结共享内存是进程间通信方式中效率最高的,原因在于进程是直接在物理内存上进行操 作,将物理地址映射到用户进程这,所以只要对其地址进行操作,就是直接对物理地址操作

二、共享内存操作

在ubuntu 12.04中共享内存限制值如下

1、共享存储区的最小字节数:1

2、共享存储区的最大字节数:32M

3、共享存储区的最大个数:4096

4、每个进程最多能映射的共享存储区的个数:4096

2.1 获得一个共享存储标识符

#include <sys/ipc.h>
#include <sys/shm.h>
 int shmget(key_t key, size_t size, int shmflg);
 功能:
     创建一个共享内存
 参数:
     key:键值,唯一的键值确定唯一的共享内存
     size:创建的共享内存的大小
     shmflg:共享内存的访问权限, 一般为 IPC_CREAT | 0777
 返回值:
     成功:共享内存的id
     失败:‐1

使用shell命令操作共享内存:

查看共享内存

         ipcs ‐m

 删除共享内存

         ipcrm ‐m shmid

 案例

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/shm.h>

int main(int argc, char const *argv[])
{
    //使用ftok函数获取键值
    key_t mykey;
    if((mykey = ftok(".", 100)) == -1)
    {
        perror("fail to ftok");
        exit(1);
    }
    //通过shmget函数创建或者打开一个共享内存,返回一个共享内存的标识符
    int shmid;
    if((shmid = shmget(mykey, 500, IPC_CREAT | 0666)) == -1)
    {
        perror("fail to shmget");
        exit(1);
    }
    printf("shmid = %d\n", shmid);
    system("ipcs -m");
    return 0;
}

执行结果

 2.2 共享内存映射(attach)

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

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

  功能:
    映射共享内存

  参数:
    shmid:共享内存的id
    shmaddr:映射的地址,设置为NULL为系统自动分配
    shmflg:标志位
        0:共享内存具有可读可写权限。
        SHM_RDONLY:只读。
 
 返回值:
     成功:映射的地址
     失败:‐1

注意 shmat函数使用的时候第二个和第三个参数一般设为NULL和0,即系统自动指定共享 内存地址,并且共享内存可读可写。

2.3 解除共享内存映射(detach)

 #include <sys/types.h>
 #include <sys/shm.h>
  int shmdt(const void *shmaddr);
 功能:
     解除共享内存的映射
 参数:
     shmaddr:映射的地址,shmat的返回值
 返回值:
     成功:0
     失败:‐1

2.4 案例:使用共享内存实现读写操作

write.c

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <string.h>

typedef struct{
    int a;
    char b;
}MSG;

int main(int argc, char const *argv[])
{
    //使用ftok函数获取键值
    key_t mykey;
    if((mykey = ftok(".", 100)) == -1)
    {
        perror("fail to ftok");
        exit(1);
    }

    //通过shmget函数创建或者打开一个共享内存,返回一个共享内存的标识符
    int shmid;
    if((shmid = shmget(mykey, 500, IPC_CREAT | 0666)) == -1)
    {
        perror("fail to shmget");
        exit(1);
    }

    system("ipcs -m");

    //使用shmat函数映射共享内存的地址
    MSG *text;
    if((text = shmat(shmid, NULL, 0)) == (void *)-1)
    {
        perror("fail to shmat");
        exit(1);
    }

    //通过shmat的返回值对共享内存操作
    text->a = 100;
    text->b = 'w';

    //操作完毕后要接触共享内存的映射
    if(shmdt(text) == -1)
    {
        perror("fail to shmdt");
        exit(1);
    }

    system("ipcs -m");
    
    return 0;
}

read.c

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/shm.h>

typedef struct{
    int a;
    char b;
}MSG;

int main(int argc, char const *argv[])
{
    //使用ftok函数获取键值
    key_t mykey;
    if((mykey = ftok(".", 100)) == -1)
    {
        perror("fail to ftok");
        exit(1);
    }

    //通过shmget函数创建或者打开一个共享内存,返回一个共享内存的标识符
    int shmid;
    if((shmid = shmget(mykey, 500, IPC_CREAT | 0666)) == -1)
    {
        perror("fail to shmget");
        exit(1);
    }

    system("ipcs -m");

    //映射共享内存的地址
    MSG *text;
    if((text = shmat(shmid, NULL, 0)) == (void *)-1)
    {
        perror("fail to shmat");
        exit(1);
    }

    //获取共享内存中的数据
    printf("a = %d, b = %c\n", text->a, text->b);

    //解除共享内存映射
    if(shmdt(text) == -1)
    {
        perror("fail to shmdt");
        exit(1);
    }

    system("ipcs -m");
    
    return 0;
}

2.5 共享内存控制

 #include <sys/ipc.h>
 #include <sys/shm.h>
 int shmctl(int shmid, int cmd, struct shmid_ds *buf);
 功能:
     设置或者获取共享内存你的属性
 参数:
     shmid:共享内存的id
     cmd:执行操作的命令
     IPC_STAT 获取共享内存的属性
     IPC_SET 设置共享内存的属性
     IPC_RMID 删除共享内存
     shmid_ds:共享内存的属性结构体
返回值:
     成功:0
     失败:‐1

案例

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/shm.h>

int main(int argc, char const *argv[])
{
    //使用ftok函数获取键值
    key_t mykey;
    if((mykey = ftok(".", 100)) == -1)
    {
        perror("fail to ftok");
        exit(1);
    }

    //通过shmget函数创建或者打开一个共享内存,返回一个共享内存的标识符
    int shmid;
    if((shmid = shmget(mykey, 500, IPC_CREAT | 0666)) == -1)
    {
        perror("fail to shmget");
        exit(1);
    }

    printf("shmid = %d\n", shmid);

    system("ipcs -m");

    //通过shmctl函数删除共享内存
    if(shmctl(shmid, IPC_RMID, NULL) == -1)
    {
        perror("fail to shmctl");
        exit(1);
    }

    system("ipcs -m");
    
    return 0;
}

执行结果

 

 最后,使用共享内存进行进程间通信时,需要注意以下几点

  1. 共享内存的大小限制:共享内存的大小通常受到系统的限制,因此在创建共享内存时需要注意不要超过系统的限制。

  2. 共享内存的访问权限:在创建共享内存时,需要指定共享内存的访问权限。不同的进程需要具有适当的权限才能访问共享内存。

  3. 共享内存的映射和解除映射:在使用共享内存时,需要使用shmat函数将共享内存映射到当前进程的地址空间,以便进行读写操作。使用完毕后,需要使用shmdt函数解除映射,以避免内存泄漏。

  4. 共享内存的竞争条件:当多个进程同时访问共享内存时,可能会发生竞争条件,导致数据不一致或程序崩溃。因此,需要使用适当的同步机制,如信号量或互斥锁,来确保对共享内存的访问是互斥的。

  • 22
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值