Linux 高级编程 - 共享内存 Shared Memory

共享内存 Shared Memory

这次我们来学习在 Linux 中最快的一种 IPC 方式:共享内存 Shared Memory,它的基本原理是:内核开辟一片内存区域,然后多个用户进程可以将这片区域映射到它们自己的地址空间中进行读写。为什么这种方式最快?因为数据不需要在进程之间复制,只要一个进程写入数据,另一个进程就能马上读取数据了,但是读取和写入必须同步

共享内存属性

我们可以在终端中使用 ipcs -m 查看系统当前开辟的共享内存:

ipcs -m

------ Shared Memory Segments --------
key        shmid      owner      perms      bytes      nattch     status 
0x00000000 104529920  orange     600        2097152    2          dest
0x00000000 688129     orange     600        524288     2          dest
0x00000000 113672194  orange     600        524288     2          dest
...

可以看到列出了共享内存的一些属性:地址,id,拥有者等等。

共享内存在哪里?

进程可以将共享内存映射到自己的地址空间中,如下:

进程地址空间

可以看到中间有一个内存映射区域,进程把内核的共享内存映射到这个地方,模型如下:

共享内存

进程可以通过共享内存的 API 获得内核所管理的共享内存的一份映射,例如直接同步读写自己的映射文件就可以实现进程 A 和 B 之间的通信了,非常简单。

共享内存的操作

共享内存使用起来比较简单,但是需要注意必须保持不同进程的读写同步,可以使用信号量或者互斥锁等,但是本篇文章主要介绍共享内存,不会涉及同步相关的操作,下面的例子主要介绍共享内存相关的 API 操作,当你熟悉了基本的用法,在以后学习同步时可以加上同步机制来保证读写正确。

共享内存(SHM)的操作主要分为下面 4 个步骤:
1. 创建或获取 SHM
2. 映射 SHM 到进程地址空间
3. 操作映射后的区域,即读写
4. 关闭进程地址空间的映射区域

分别来了解这些操作的 API。

1. 创建:shmge

  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Linux共享内存编程是指在Linux操作系统中使用共享内存实现进程间通信编程技术。共享内存是一种特殊的内存区域,它可以被多个进程同时访问,从而实现进程间数据的共享。 在Linux中,使用共享内存需要经过以下几个步骤: 1. 创建共享内存区域:使用shmget()函数创建一个共享内存区域,并返回该区域的标识符。 2. 连接共享内存区域:使用shmat()函数将进程与共享内存区域连接起来,并返回连接后的内存地址。 3. 使用共享内存:进程可以使用连接后的内存地址读写共享内存中的数据。 4. 分离共享内存区域:使用shmdt()函数将进程与共享内存区域分离,使得进程不能再访问该内存区域。 5. 删除共享内存区域:使用shmctl()函数删除共享内存区域。 下面是一个使用共享内存实现进程间通信的例子: ``` #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys/ipc.h> #include <sys/shm.h> #define SHM_SIZE 1024 int main() { int shmid; key_t key; char *shm, *s; // 生成共享内存key值 key = ftok(".", 'a'); // 创建共享内存区域 if ((shmid = shmget(key, SHM_SIZE, IPC_CREAT | 0666)) < 0) { perror("shmget"); exit(1); } // 连接共享内存区域 if ((shm = shmat(shmid, NULL, 0)) == (char *) -1) { perror("shmat"); exit(1); } // 写入共享内存数据 s = shm; for (char c = 'a'; c <= 'z'; c++) { *s++ = c; } *s = '\0'; // 分离共享内存区域 if (shmdt(shm) == -1) { perror("shmdt"); exit(1); } // 重新连接共享内存区域 if ((shm = shmat(shmid, NULL, 0)) == (char *) -1) { perror("shmat"); exit(1); } // 读取共享内存数据 printf("Shared memory content: "); for (s = shm; *s != '\0'; s++) { putchar(*s); } putchar('\n'); // 分离共享内存区域 if (shmdt(shm) == -1) { perror("shmdt"); exit(1); } // 删除共享内存区域 if (shmctl(shmid, IPC_RMID, 0) == -1) { perror("shmctl"); exit(1); } return 0; } ``` 这个例子中,程序通过ftok()函数生成一个共享内存的key值,然后使用shmget()函数创建共享内存区域,使用shmat()函数将进程与共享内存区域连接起来,然后向共享内存中写入数据,再使用shmdt()函数分离共享内存区域,重新连接共享内存区域并读取数据,最后使用shmctl()函数删除共享内存区域。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值