IPC之共享内存
man 7 shm_overview
shm_overview - Overview of POSIX shared memory.
同样,SystemV实现的共享内存是旧的机制,但应用广泛;Posix标准提供了新的统一接口。
共享内存是由内核出于在多个进程间交换信息的目的而留出的一块内存区(段)。如果段的权限设置恰当,每个要访问该段内存的进程都可以把它映射到自己私有的地址空间中。如果一个进程更新了段中数据,那么其他进程立即回看到更新。由一个进程创建的段也可以由另一个进程读写。共享内存这一名称表达出是由多个进程分享对段及其保存的数据的访问权这一含义。共享内存很像内存映射文件。
Posix API:
shm_open, ftruncate, mmap, munmap, shm_unlink, close, fstat, fchown, fchmod.
System V API:
shmget,shmat,shmdt
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
int shmget(key_t key, int size, int flags);
flags可以是一个或多个IPC_CREAT、IPC_EXCL和一组权限位(模式)按位“或”的结果。权限位以八进制表示。IPC_EXCL确保如果段已经存在,执行失败,而不是返回一个已经存在的段的标示符。IPC_CREAT指出如果没有和key关联的段就应该创建一个新段。key既可以是IPC_PRIVATE也可以是ftok函数返回的一个关键字。参数size指定段的大小,但它以PAGE_SIZE的值为界,这个值是某种处理器本身页的大小(intel是4KB)。shmget成功返回段标识符,出错返回-1。
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <stdio.h>
#include <stdlib.h> #define BUFSZ 4096 int main(void)
{
int shmid; if((shmid = shmget(IPC_PRIVATE, BUFSZ, )) < )
{
perror("shmget");
exit(EXIT_FAILURE);
} printf("segment created: %d\n", shmid); system("ipcs -m\n"); exit(EXIT_SUCCESS);
}
~$./a.out
segment created: ------ Shared Memory Segments --------
key shmid owner perms bytes nattch status
0x00000000 yuxi
0x00000000 yuxi
0x00000000 yuxi
shmget只是创建了共享内存区,进程要把它映射到自己的地址空间才能使用它,调用shmat完成。
#include <sys/types.h>
#include <sys/shm.h>
void *shmat(int shmid, const char *shmaddr, int shmflg);
int shmdt(const void *shmaddr);
在函数shmat中,如果shmaddr为NULL,则内核会把段映射到调用进程的地址空间中它所选定的位置。假如shmaddr不为NULL,并且shmflg指定SHM_RND,则attach发生在附近的最小倍数SHMLBA。否则shmaddr必须是页对齐的地址。一般总是把shmaddr设置为0。flags可以为SHM_RDONLY,这意味着被附加的段是只读的。否则,被附加的段默认是可读写的。如果shmat调用成功,则返回附加了段的地址。否则,它返回-1并且设置errno变量。
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/shm.h> int main(int argc, char *argv[])
{
int shmid;
char *shmbuf; if(argc != )
{
puts("USAGE: atshm <identifier>");
exit(EXIT_FAILURE);
}
shmid = atoi(argv[]); if((shmbuf = shmat(shmid, , )) < (char *))
{
perror("shmat");
exit(EXIT_FAILURE);
}
printf("segment attached at %p\n", shmbuf); system("ipcs -m"); if((shmdt(shmbuf)) < )
{
perror("shmdt");
exit(EXIT_FAILURE);
} puts("segment detached"); system("ipcs -m"); exit(EXIT_SUCCESS);
}
~$./a.out
segment attached at 0xb7709000 ------ Shared Memory Segments --------
key shmid owner perms bytes nattch status
0x00000000 yuxi
0x00000000 yuxi
0x00000000 yuxi segment detached ------ Shared Memory Segments --------
key shmid owner perms bytes nattch status
0x00000000 yuxi
0x00000000 yuxi
0x00000000 yuxi
使用举例:
include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/shm.h>
#include <sys/ipc.h>
#include <fcntl.h>
#include <ctype.h> #define BUFSZ 4096 int main(int argc, char **argv)
{
int shmid;
char *shmbuf;
int fd;
int i; if(argc != )
{
puts("USAGE: opshm <identifier>");
exit(EXIT_FAILURE);
}
shmid = atoi(argv[]); if((shmbuf = shmat(shmid, , )) < (char *))
{
perror("shmat");
exit(EXIT_FAILURE);
} if((shmbuf = malloc(sizeof(char) * BUFSZ)) < (char *))
{
perror("malloc");
exit(EXIT_FAILURE);
} for(i = ; i < BUFSZ; ++i)
{
shmbuf[i] = rand();
} fd = open("opshm.out", O_CREAT | O_WRONLY, );
write(fd, shmbuf, BUFSZ); free(shmbuf); exit(EXIT_SUCCESS);
}
创建shm举例
void *init_shm(void)
{
key_t key;
int shmid;
void *addr = NULL; key = ftok(SHM_PATH, SHM_PROJ);
if(key == -)
{
perror("ftok()");
return NULL;
} shmid = shmget(key, sizeof(struct shm_block), IPC_CREAT | );
if(shmid == -)
{
perror("shmget: ");
return NULL;
}
addr = shmat(shmid, , );
if (addr == (void *)-)
{
perror("shmat()");
return NULL;
} return addr;
}
通过命令ipcs -m可以查看共享内存信息。
注:大部分内容来自《GNU/LINUX编程指南》
IPC之共享内存的更多相关文章
- 进程间通信IPC之--共享内存
每个进程各自有不同的用户地址空间,任何一个进 程的全局变量在另一个进程中都看不到,所以进程之间要交换数据必须通过内核,在内核中开辟一块缓冲 区,进程1把数据从用户空间拷到内核缓冲区,进程2再从内核缓冲 ...
- Linux IPC之共享内存C 事例
Linux IPC之共享内存 标签: linuxrandomnull工作 2011-08-25 11:52 4123人阅读 评论(0) 收藏 举报 分类: Linux(3) 读书札记(3) 版权 ...
- System V IPC(3)-共享内存
一.概述 1.共享内存允许多个进程共享物理内存的同一块内存区. 2.与管道和消息队列不同,共享内存 ...
- System V IPC 之共享内存
IPC 是进程间通信(Interprocess Communication)的缩写,通常指允许用户态进程执行系列操作的一组机制: 通过信号量与其他进程进行同步 向其他进程发送消息或者从其他进程接收消息 ...
- 进程间通信——IPC之共享内存
共享内存是三个IPC机制中的一个.它允许两个不相关的进程访问同一个逻辑内存.共享内存是在两个正在进行的进程之间传递数据的一种非常有效的方式. 大多数的共享内存的实现,都把由不同进程之间共享 ...
- 五十、进程间通信——System V IPC 之共享内存
50.1 共享内存 50.1.1 共享内存的概念 共享内存区域是被多个进程共享的一部分物理内存 多个进程都可把该共享内存映射到自己的虚拟内存空间.所有用户空间的进程若要操作共享内存,都要将其映射到自己 ...
- zabbix_agentd重装后启动时IPC和共享内存段问题
zabbix_agentd不知为啥被干掉后重装了zabbix,zabbix用户组id也变了. 重装zabbix后导致zabbix_agentd无法启动,两个问题 问题1: zabbix_agentd ...
- Linux IPC之共享内存
System V共享内存机制: shmget shmat shmdt shmctl 原理及实现: system V IPC机制下的共享内存本质是一段特殊的内存区域,进程间需要共享的数据被放在该共 ...
- Unix IPC之共享内存区(1)
1 共享内存区 共享内存区是可用IPC形式中最快的,只有映射和解除映射需要进入内核的系统调用,映射后对共享内存区的访问和修改不再需要系统调用(内核只要负责好页表映射和处理页面故障即可),但通常需要同步 ...
- System IPC 与Posix IPC(共享内存)
系统v(共享内存) 1.对于系统V共享内存,主要有以下几个API:shmget().shmat().shmdt()及shmctl(). 2.shmget()用来获得共享内存区域的ID,如果不存在指定的 ...
随机推荐
- screen space reflection/soft alpha test/
http://www.crytek.com/cryengine/presentations/secrets-of-cryengine-3-graphics-technology 很多宝贝里面 不止题目 ...
- C++ 初始化列表(转载)
何谓初始化列表 与其他函数不同,构造函数除了有名字,参数列表和函数体之外,还可以有初始化列表,初始化列表以冒号开头,后跟一系列以逗号分隔的初始化字段.在 C++中,struct和class的唯一区别是 ...
- CCControlExtension/CCControlButton
#ifndef __CCCONTROL_BUTTON_H__ #define __CCCONTROL_BUTTON_H__ #include "CCControl.h" #incl ...
- Maven+TestNG+ReportNG/Allure接口自动化测试框架初探(上)
转载:http://www.51testing.com/html/58/n-3721258.html 由于一直忙于功能和性能测试,接口自动化测试框架改造的工作被耽搁了好久.近期闲暇一些,可以来做点有意 ...
- TestNG 八 并发测试
一. Concurrenttesting: 下面的例子是输出进程ID,threadPoolSize用来指明线程池的大小,也就是并发的线程数目是多少 5次调用,有3个线程可调用 @Test(invoca ...
- B. Vanya and Books( Codeforces Round #308 (Div. 2) 简单题)
B. Vanya and Books time limit per test 1 second memory limit per test 256 megabytes input standard i ...
- Android 文件存储 和 权限管理
转载请标明出处: :http://blog.csdn.net/huaiyiheyuan/article/details/52473984 android SD卡主要有两种存储方式 Internal . ...
- java Web笔记
Get:标记在地址栏中 放在请求行中显示get?Username=...Password=... Post:使用的是二进制的上传下载 请求行不显示信息 数据传输量大数据隐蔽对文本没有 pst不能再浏览 ...
- STL源码剖析(算法)
STL中算法是基于迭代器来实现的. 有了容器中迭代器的实现(对operator*.operator++等的重载),STL中大部分算法实现就显得很简单了. 先看一例关于find算法的实现: templa ...
- hibernate 多对一关联
(转自尚学堂教学内容) 注解多对一: package com.bjsxt.hibernate; import javax.persistence.Entity; import javax.pers ...