Linux IPC实践(10) --Posix共享内存

简介: 1. 创建/获取一个共享内存#include #include /* For mode constants */#include /* For O...

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

目录
相关文章
|
2月前
|
传感器 数据采集 监控
Python生成器与迭代器:从内存优化到协程调度的深度实践
简介:本文深入解析Python迭代器与生成器的原理及应用,涵盖内存优化技巧、底层协议实现、生成器通信机制及异步编程场景。通过实例讲解如何高效处理大文件、构建数据流水线,并对比不同迭代方式的性能特点,助你编写低内存、高效率的Python代码。
127 0
|
2月前
|
边缘计算 算法 Java
Java 绿色计算与性能优化:从内存管理到能耗降低的全方位优化策略与实践技巧
本文探讨了Java绿色计算与性能优化的技术方案和应用实例。文章从JVM调优(包括垃圾回收器选择、内存管理和并发优化)、代码优化(数据结构选择、对象创建和I/O操作优化)等方面提出优化策略,并结合电商平台、社交平台和智能工厂的实际案例,展示了通过Java新特性提升性能、降低能耗的显著效果。最终指出,综合运用这些优化方法不仅能提高系统性能,还能实现绿色计算目标,为企业节省成本并符合环保要求。
90 0
|
3月前
|
SQL 缓存 安全
深度理解 Java 内存模型:从并发基石到实践应用
本文深入解析 Java 内存模型(JMM),涵盖其在并发编程中的核心作用与实践应用。内容包括 JMM 解决的可见性、原子性和有序性问题,线程与内存的交互机制,volatile、synchronized 和 happens-before 等关键机制的使用,以及在单例模式、线程通信等场景中的实战案例。同时,还介绍了常见并发 Bug 的排查与解决方案,帮助开发者写出高效、线程安全的 Java 程序。
172 0
|
5月前
|
缓存 Linux 数据安全/隐私保护
Linux环境下如何通过手动调用drop_caches命令释放内存
总的来说,记录住“drop_caches” 命令并理解其含义,可以让你在日常使用Linux的过程中更加娴熟和自如。
1028 23
|
7月前
|
监控 Linux Python
Linux系统资源管理:多角度查看内存使用情况。
要知道,透过内存管理的窗口,我们可以洞察到Linux系统运行的真实身姿,如同解剖学家透过微观镜,洞察生命的奥秘。记住,不要惧怕那些高深的命令和参数,他们只是你掌握系统"魔法棒"的钥匙,熟练掌握后,你就可以骄傲地说:Linux,我来了!
235 27
|
8月前
|
消息中间件 Linux
Linux中的System V通信标准--共享内存、消息队列以及信号量
希望本文能帮助您更好地理解和应用System V IPC机制,构建高效的Linux应用程序。
278 48
|
8月前
|
缓存 NoSQL Linux
Linux系统内存使用优化技巧
交换空间(Swap)的优化 禁用 Swap sudo swapoff -a 作用:这个命令会禁用系统中所有的 Swap 空间。swapoff 命令用于关闭 Swap 空间,-a 参数表示关闭 /etc/fstab 文件中配置的所有 Swap 空间。 使用场景:在高性能应用场景下,比如数据库服务器或高性能计算服务器,禁用 Swap 可以减少磁盘 I/O,提高系统性能。
294 3
|
8月前
|
缓存 Linux
Linux查看内存命令
1. free free命令是最常用的查看内存使用情况的命令。它显示系统的总内存、已使用内存、空闲内存和交换内存的总量。 free -h • -h 选项:以易读的格式(如GB、MB)显示内存大小。 输出示例: total used free shared buff/cache available Mem: 15Gi 4.7Gi 4.1Gi 288Mi 6.6Gi 9.9Gi Swap: 2.0Gi 0B 2.0Gi • to
578 2
|
3月前
|
存储
阿里云轻量应用服务器收费标准价格表:200Mbps带宽、CPU内存及存储配置详解
阿里云香港轻量应用服务器,200Mbps带宽,免备案,支持多IP及国际线路,月租25元起,年付享8.5折优惠,适用于网站、应用等多种场景。
835 0
|
3月前
|
存储 缓存 NoSQL
内存管理基础:数据结构的存储方式
数据结构在内存中的存储方式主要包括连续存储、链式存储、索引存储和散列存储。连续存储如数组,数据元素按顺序连续存放,访问速度快但扩展性差;链式存储如链表,通过指针连接分散的节点,便于插入删除但访问效率低;索引存储通过索引表提高查找效率,常用于数据库系统;散列存储如哈希表,通过哈希函数实现快速存取,但需处理冲突。不同场景下应根据访问模式、数据规模和操作频率选择合适的存储结构,甚至结合多种方式以达到最优性能。掌握这些存储机制是构建高效程序和理解高级数据结构的基础。
239 0

热门文章

最新文章