Linux IPC实践(10) --Posix共享内存
1. 创建/获取一个共享内存
#include <sys/mman.h> #include <sys/stat.h> /* For mode constants */ #include <fcntl.h> /* For O_* constants */ int shm_open(const char *name, int oflag, mode_t mode);
参数:
name: 共享内存名字;
oflag: 与open函数类型, 可以是O_RDONLY, O_WRONLY, O_RDWR, 还可以按位或上O_CREAT, O_EXCL, O_TRUNC.
mode: 此参数总是需要设置, 如果oflag没有指定O_CREAT, 则mode可以设置为0;
返回值:
成功: 返回一个文件描述符;
失败: 返回-1;
注意-Posix IPC名字限制:
1. 必须以”/”开头, 并且后面不能还有”/”, 形如:/file-name;
2. 名字长度不能超过NAME_MAX
3. 链接时:Link with -lrt.
/** 示例: 共享内存的打开与关闭 **/
int main(int argc,char *argv[])
{
int shmid = shm_open("/xyz", O_RDWR|O_CREAT, 0666);
if (shmid == -1)
err_exit("shmget error");
cout << "share memory open success" << endl;
close(shmid);
}
2. 修改共享内存大小
int ftruncate(int fd, off_t length);
该函数不仅可用于修改共享内存大小, 而且可以用于修改文件大小
/** 示例: 修改共享内存大小
将其修改为一个Student结构体的大小
**/
struct Student
{
char name[32];
int age;
};
int main(int argc,char *argv[])
{
int shmid = shm_open("/xyz", O_RDWR|O_CREAT, 0666);
if (shmid == -1)
err_exit("shmget error");
if (ftruncate(shmid, sizeof(Student)) == -1)
err_exit("ftruncate error");
cout << "share memory change size success" << endl;
close(shmid);
}
3. 获取共享内存状态
int fstat(int fd, struct stat *buf);
该函数不仅可用于获取共享内存状态, 而且可以用于获取文件状态, 与前面曾经讲述过的stat, lstat类型;
//stat结构体
struct stat
{
dev_t st_dev; /* ID of device containing file */
ino_t st_ino; /* inode number */
mode_t st_mode; /* protection */
nlink_t st_nlink; /* number of hard links */
uid_t st_uid; /* user ID of owner */
gid_t st_gid; /* group ID of owner */
dev_t st_rdev; /* device ID (if special file) */
off_t st_size; /* total size, in bytes */
blksize_t st_blksize; /* blocksize for filesystem I/O */
blkcnt_t st_blocks; /* number of 512B blocks allocated */
time_t st_atime; /* time of last access */
time_t st_mtime; /* time of last modification */
time_t st_ctime; /* time of last status change */
};
/** 示例: 获取共享内存的mode和size **/
int main(int argc,char *argv[])
{
int shmid = shm_open("/xyz", O_RDWR|O_CREAT, 0666);
if (shmid == -1)
err_exit("shmget error");
if (ftruncate(shmid, sizeof(Student)) == -1)
err_exit("ftruncate error");
struct stat buf;
if (fstat(shmid, &buf) == -1)
err_exit("lstat error");
// 注意: 获取权限时, 需要&上0777, 而且要以%o, 八进制方式打印
printf("mode: %o\n", buf.st_mode&0777);
printf("size: %ld\n", buf.st_size);
close(shmid);
}
4. 删除一个共享内存对象
int shm_unlink(const char *name);
//示例:
int main(int argc,char *argv[])
{
shm_unlink("/xyz");
}
5. 共享内存的映射/卸载
#include <sys/mman.h> void *mmap(void *addr, size_t length, int prot, int flags, int fd, off_t offset); int munmap(void *addr, size_t length);
参数:
addr: 要映射的起始地址, 通常指定为NULL, 让内核自动选择;
length: 映射到进程地址空间的字节数, 通常是先前已经创建的共享内存的大小;
prot: 映射区保护方式(见下);
flags: 标志(通常指定为MAP_SHARED, 用于进程间通信);
fd: 文件描述符(填为shm_open返回的共享内存ID);
offset: 从文件头开始的偏移量(一般填为0);
|
prot |
说明 |
|
PROT_READ |
页面可读 |
|
PROT_WRITE |
页面可写 |
|
PROC_EXEC |
页面可执行 |
|
PROC_NONE |
页面不可访问 |
|
flags |
说明 |
|
MAP_SHARED |
变动是共享的 |
|
MAP_PRIVATE |
变动是私有的 |
|
MAP_FIXED |
准确解释addr参数, 如果不指定该参数, 则会以4K大小的内存进行对齐 |
|
MAP_ANONYMOUS |
建立匿名映射区, 不涉及文件 |
mmap返回值:
成功: 返回映射到的内存区的起始地址;
失败: 返回MAP_FAILED;
注意:mmap失败返回EACCES错误的原因:
EACCES A file descriptor refers to a non-regular file.
Or MAP_PRIVATE was requested, but fd is not open for reading.
Or MAP_SHARED was requested and PROT_WRITE is set, but fd is not open
in read/write (O_RDWR) mode. Or PROT_WRITE is set, but the file is append-only.
/** 示例: 向共享内存写入数据 **/
int main(int argc,char *argv[])
{
int shmid = shm_open("/xyz", O_RDWR, 0);
if (shmid == -1)
err_exit("shm_open error");
struct stat buf;
if (fstat(shmid, &buf) == -1)
err_exit("fstat error");
Student *p = (Student *)mmap(NULL, buf.st_size,
PROT_WRITE, MAP_SHARED, shmid, 0);
if (p == MAP_FAILED)
err_exit("mmap error");
strcpy(p->name, "Hadoop");
p->age = 5;
munmap(p, buf.st_size);
close(shmid);
}
/** 从共享内存读出数据 **/
int main(int argc,char *argv[])
{
int shmid = shm_open("/xyz", O_RDONLY, 0);
if (shmid == -1)
err_exit("shm_open error");
struct stat buf;
if (fstat(shmid, &buf) == -1)
err_exit("fstat error");
Student *p = (Student *)mmap(NULL, buf.st_size,
PROT_READ, MAP_SHARED, shmid, 0);
if (p == MAP_FAILED)
err_exit("mmap error");
cout << "name: " << p->name << ", age: " << p->age << endl;
munmap(p, buf.st_size);
close(shmid);
}
[附]
-Posix共享内存默认创建在/dev/shm目录下
-可以使用od命令查看共享内存的内容
od -c /dev/shm/xyz
Linux IPC实践(10) --Posix共享内存的更多相关文章
- linux c编程:Posix共享内存区
Posix共享内存区:共享内存是最快的可用IPC形式.它允许多个不相关(无亲缘关系)的进程去访问同一部分逻辑内存.如果需要在两个进程之间传输数据,共享内存将是一种效率极高的解决方案.一旦这样的内存区映 ...
- linux网络编程之posix共享内存
今天继续研究posix IPC对象,这次主要是学习一下posix共享内存的使用方法,下面开始: 下面编写程序来创建一个共享内存: 编译运行: 那posix的共享内存存放在哪里呢?上节中学的posix的 ...
- Linux IPC实践(7) --Posix消息队列
1. 创建/获取一个消息队列 #include <fcntl.h> /* For O_* constants */ #include <sys/stat.h> /* For m ...
- 细说linux IPC(四):posix 共享内存
[版权声明:尊重原创.转载请保留出处:blog.csdn.net/shallnet 或 .../gentleliu,文章仅供学习交流,请勿用于商业用途] 上一节讲了由open函数打开一 ...
- Linux进程IPC浅析[进程间通信SystemV共享内存]
Linux进程IPC浅析[进程间通信SystemV共享内存] 共享内存概念,概述 共享内存的相关函数 共享内存概念,概述: 共享内存区域是被多个进程共享的一部分物理内存 多个进程都可把该共享内存映射到 ...
- Posix共享内存区
1.概述 Posix提供了两种在无亲缘关系进程间共享内存区的方法: (1)内存映射文件:先有open函数打开,然后调用mmap函数把得到的描述符映射到当前进程地址空间中的一个文件(上一篇笔记所用到的就 ...
- POSIX共享内存
DESCRIPTION 共享内存是最快的可用IPC形式.它允许多个不相关(无亲缘关系)的进程去访问同一部分逻辑内存. 如果需要在两个进程之间传输数据,共享内存将是一种效率极高的解决方案.一旦这样的内存 ...
- POSIX 共享内存和 系列函数
在前面介绍了system v 共享内存的相关知识,现在来稍微看看posix 共享内存 和系列函数. 共享内存简单来说就是一块真正的物理内存区域,可以使用一些函数将这块区域映射到进程的地址空间进行读写, ...
- 第三十五章 POSIX共享内存
POSIX共享内存函数介绍 shm_open 功能: 用来创建或打开一个共享内存对象 原型: int shm_open(const char *name, int oflag, mode_t mode ...
随机推荐
- javaweb面试题
1.Tomcat的优化经验 答:去掉对web.xml的监视,把JSP提前编辑成Servlet:有富余物理内存的情况下,加大Tomcat使用的JVM内存. 2.什么是Servlet? 答:可以从两个方面 ...
- virtualenvwrapper 的安装和使用
virtualenvwrapper是用来管理virtualenv的扩展包,用着很方便. 1. 安装: #安装virtualenv (sudo) pip install virtualenv #安装vi ...
- Linux搭建lamp(Apache+PHP+Mysql环境)centos7.2版详细教程
我们更多的网站服务器是选择了Linux系统,这里建议你选择centos,这也是阿里云ecs推荐的系统,在服务器上比较推荐centos,特别对于新手,首选CentOS,并不是centos比Debian和 ...
- mysql5.7在centos上安装的完整教程以及相关的“坑”
安装前的准备 Step1: 如果你系统已经有mysql,如一般centos自带mysql5.1系列,那么你需要删除它,先检查一下系统是否自带mysql yum list installed | gre ...
- chrome官方完整安装包
But did you know Google allows you to download the full standalone installer of Chrome from its offi ...
- Kafka系列之-自定义Producer
前面已经讲到了,在Kafka中,Message是由Producer产生的,Producer产生的Message会发送到Topic的指定Partition中.Producer可以有多种形式,也可以由用户 ...
- Android Studio精彩案例(二)《仿微信动态点击底部tab切换Fragment》
转载本专栏文章,请注明出处,尊重原创 .文章博客地址:道龙的博客 现在很多的App要么顶部带有tab,要么就底部带有tab.用户通过点击tab从而切换不同的页面(大部分情况时去切换fragment). ...
- Linux文件格式化与相关处理及sed工具
http://blog.csdn.net/pipisorry/article/details/52564957 geditor, sed, ed, awk, Emacs 文本处理工具 [Linux S ...
- 微信小程序基础之常用控件text、icon、progress、button、navigator
今天展示一下基础控件的学习开发,希望对大家有所帮助,转载请说明~ 首先延续之前的首页界面展示,几个跳转navigator的使用,然后是各功能模块的功能使用 一.text展示 使用按钮,进行文字的添加与 ...
- 用户创建,删除and并发注册and系统登陆的API研究(学习汇总网上资料)
一.系统登陆链接实现 比如有一个外围支持系统,用户需要在外围系统登录之后点个link就可以登录到Oracle ERP系统中,那么我们需要先把外围系统的用户创建在Oracle ERP中,并且分配职责给他 ...