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月前
|
安全 Linux Shell
Linux上执行内存中的脚本和程序
【9月更文挑战第3天】在 Linux 系统中,可以通过多种方式执行内存中的脚本和程序:一是使用 `eval` 命令直接执行内存中的脚本内容;二是利用管道将脚本内容传递给 `bash` 解释器执行;三是将编译好的程序复制到 `/dev/shm` 并执行。这些方法虽便捷,但也需谨慎操作以避免安全风险。
152 6
|
10天前
|
监控 Linux 云计算
Linux操作系统在云计算环境中的实践与优化###
【10月更文挑战第16天】 本文探讨了Linux操作系统在云计算环境中的应用实践,重点分析了其在稳定性、安全性和高效性方面的优势。通过具体案例,阐述了Linux如何支持虚拟化技术、实现资源高效分配以及与其他开源技术的无缝集成。文章还提供了针对Linux系统在云计算中的优化建议,包括内核参数调整、文件系统选择和性能监控工具的应用,旨在帮助读者更好地理解和应用Linux于云计算场景。 ###
13 3
|
19天前
|
存储 缓存 监控
Linux中内存和性能问题
【10月更文挑战第5天】
33 4
|
18天前
|
算法 Linux
Linux中内存问题
【10月更文挑战第6天】
17 2
|
19天前
|
存储 缓存 固态存储
|
28天前
|
Ubuntu Linux
Linux实践|设置静态 IP 地址
Linux实践|设置静态 IP 地址
45 0
Linux实践|设置静态 IP 地址
|
21天前
|
Linux C++
Linux c/c++文件虚拟内存映射
这篇文章介绍了在Linux环境下,如何使用虚拟内存映射技术来提高文件读写的速度,并通过C/C++代码示例展示了文件映射的整个流程。
34 0
|
24天前
|
SQL 安全 算法
ChatGPT高效提问—prompt实践(漏洞风险分析-重构建议-识别内存泄漏)
ChatGPT高效提问—prompt实践(漏洞风险分析-重构建议-识别内存泄漏)
30 0
|
27天前
|
程序员 编译器 数据处理
【C语言】深度解析:动态内存管理的机制与实践
【C语言】深度解析:动态内存管理的机制与实践
|
3月前
|
数据采集 Rust 安全
Rust在网络爬虫中的应用与实践:探索内存安全与并发处理的奥秘
【8月更文挑战第31天】网络爬虫是自动化程序,用于从互联网抓取数据。随着互联网的发展,构建高效、安全的爬虫成为热点。Rust语言凭借内存安全和高性能特点,在此领域展现出巨大潜力。本文探讨Rust如何通过所有权、借用及生命周期机制保障内存安全;利用`async/await`模型和`tokio`运行时处理并发请求;借助WebAssembly技术处理动态内容;并使用`reqwest`和`js-sys`库解析CSS和JavaScript,确保代码的安全性和可维护性。未来,Rust将在网络爬虫领域扮演更重要角色。
71 1