Linux: shm_xx系列函数使用详解

2024-03-02 13:04

本文主要是介绍Linux: shm_xx系列函数使用详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

  • 一、shmget/shmctl/shmat/shmdt函数
    • 1、shmget
    • 2、shmctl
    • 3、shmat
    • 4、shmdt
    • 5、补充:ftok函数
    • 6、示例代码
  • 二、shm_open/shm_unlink函数
    • 1、shm_open
    • 2、shm_unlink
    • 3、示例代码
  • 三、课外阅读

一、shmget/shmctl/shmat/shmdt函数

  shm_xx系列函数是用于操作共享内存的一组函数,它们通常包括shmget、shmctl、shmat和shmdt。下面对这些函数进行详细解释:

1、shmget

  • 功能:创建一个新的共享内存段或获取一个已经存在的共享内存段的标识符。
  • 原型int shmget(key_t key, size_t size, int shmflg)
  • 参数
    • key:用于识别共享内存段的键值,通常使用 ftok 函数生成。
    • size:共享内存段的大小。单位是字节。
    • shmflg:打开标志,用于指定操作的方式,比如创建共享内存段、获取共享内存标识符等。
  • 返回值:成功时返回共享内存段的标识符(非负整数),失败时返回-1。
    shmget 函数是用于创建共享内存段、获取共享内存标识符或者访问一个已存在的共享内存段的函数,其声明如下:

  shmget 函数的主要功能包括:

  • 如果指定的 key 对应的共享内存段不存在,并且设置了创建标志(IPC_CREAT),则会创建一个新的共享内存段,大小为 size 字节。
  • 如果指定的 key 对应的共享内存段存在,则会返回共享内存标识符。
  • 如果不希望创建共享内存段,可以不设置 IPC_CREAT 标志。此时若指定的 key 对应的共享内存段不存在,shmget 函数将会返回错误。

  shmget 函数的返回值是共享内存标识符(shmid),用于后续对共享内存段的访问操作,比如连接共享内存、获得共享内存的地址等。

  在使用 shmget 函数时,需要注意确保对共享内存的大小和权限的设置,避免潜在的问题,比如内存溢出或者权限不足等。同时,需要根据具体需求合理设置 shmflg 参数,确保对共享内存的访问操作符合预期。

2、shmctl

  • 功能:用于控制共享内存段。
  • 原型int shmctl(int shmid, int cmd, struct shmid_ds *buf)
  • 参数
    • shmid:是共享内存标识符,是由 shmget 函数返回的。
    • cmd:是控制命令,用于指定对共享内存的具体操作,比如删除共享内存段、获取共享内存信息等。
    • buf:是指向 shmid_ds 结构体的指针,用于传递或者获取共享内存的信息。
  • 返回值:成功时返回0,失败时返回-1。
    shmctl 函数是用于控制共享内存段的操作函数,主要用于对共享内存段进行控制操作,比如删除、获取信息等,其声明如下:

shmctl 函数的主要功能包括:

  • 获取共享内存段的信息,比如共享内存大小、关联进程数等。
  • 控制共享内存段的权限,比如修改共享内存的权限。
  • 删除共享内存段。

cmd 参数可以取以下值之一:

  • IPC_STAT:获取共享内存的状态信息,并将其存储在 buf 指向的结构体中。
  • IPC_SET:设置共享内存的状态信息,更新共享内存的权限等。
  • IPC_RMID:删除共享内存段,释放资源。

  需要注意的是,在使用 shmctl 函数时,要根据具体需求选择合适的控制命令,并确保传入正确的参数,以免出现意外的错误。同时,删除共享内存段时,需要注意确保所有关联进程已经分离了共享内存段,否则删除操作可能会失败。

3、shmat

  • 功能:将共享内存段连接到调用进程的地址空间。
  • 原型void *shmat(int shmid, const void *shmaddr, int shmflg)
  • 参数
    • shmid:是共享内存标识符,是由 shmget 函数返回的。
    • shmaddr:是一个指针,用于指定将共享内存段映射到进程地址空间的地址,通常设为 NULL
    • shmflg:参数用于指定连接共享内存段的标志,通常可以设置为 0。
  • 返回值:成功时返回指向共享内存段起始地址的指针,失败时返回-1。

  shmat 函数的主要作用是将共享内存段连接到当前进程的地址空间,实现进程对共享内存段的访问。连接成功后,shmat 函数将返回一个指向共享内存段起始地址的指针,进程可以通过该指针来访问共享内存段中的数据。

  一般情况下,shmaddr 参数设为 NULL,由系统自动选择合适的地址来映射共享内存段到进程地址空间中。shmflg 参数通常设为 0,表示默认的连接标志,除非有特殊需求,一般不需要修改。

  需要注意的是,在使用 shmat 函数时,要确保传入正确的共享内存标识符 shmid,避免出现连接错误。另外,在使用完共享内存后,要及时调用 shmdt 函数将共享内存段从当前进程地址空间中分离,避免资源泄漏。

4、shmdt

  • 功能:将共享内存段从调用进程的地址空间中分离。
  • 原型int shmdt(const void *shmaddr)
  • 参数
    • shmaddr:是共享内存段连接到进程地址空间的起始地址,通常是由 shmat 函数返回的。
  • 返回值:成功时返回0,失败时返回-1。

  shmdt 函数的主要作用是断开当前进程与共享内存段的连接,使得进程无法再访问共享内存段中的数据。调用 shmdt 函数后,共享内存段仍然存在,但当前进程无法再对其进行操作。

  在使用 shmdt 函数时,需要传入正确的共享内存段起始地址 shmaddr,确保将正确的共享内存段从当前进程中分离。另外,需要注意的是,分离共享内存段后,如果没有其他进程仍然连接着该共享内存段,系统会释放该共享内存段,从而释放相关资源。

  在使用共享内存时,需要注意合理地管理共享内存的连接和分离,避免出现资源泄漏或者错误的操作。

ftok 函数是一个用于生成一个 System V IPC(Inter-Process Communication,进程间通信)键值的函数,其声明如下:

5、补充:ftok函数

key_t ftok(const char *pathname, int proj_id);
  • pathname 参数是一个指向字符串的指针,用于指定一个已经存在的文件的路径名。
  • proj_id 参数是一个用户定义的整数,可以是 0 到 255 的范围内的任意值。

  ftok 函数会根据指定的 pathnameproj_id 生成一个唯一的键值,用于创建或访问 System V IPC 的资源,如共享内存、信号量和消息队列。

  实际上,ftok 函数根据 pathname 参数对应文件的 inode 号和 proj_id 参数生成一个 32 位的键值,最高 8 位存放 proj_id,低 24 位存放 pathname 对应文件的 inode 号的后 24 位。这样可以保证相同的 pathnameproj_id 参数生成相同的键值。

  需要注意的是,在使用 ftok 函数时要确保 pathname 参数是一个已经存在的文件的路径名,否则会导致生成的键值不唯一。另外,proj_id 参数的取值范围是 0 到 255,超出这个范围可能导致键值重复。

  综上所述,ftok 函数是用于生成 System V IPC 键值的一个实用工具函数,可以通过指定文件路径和项目标识来确保生成唯一的键值,用于进程间通信的各种资源的创建和访问。

6、示例代码

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/shm.h>#define SHM_SIZE 1024 // 共享内存大小int main() {key_t key = ftok("shmfile", 'R'); // 创建共享内存的键值int shmid;// 创建共享内存段shmid = shmget(key, SHM_SIZE, IPC_CREAT | 0666);if (shmid == -1) {perror("shmget");exit(1);}// 连接共享内存段char *shmaddr = (char *)shmat(shmid, NULL, 0);if (shmaddr == (char *)-1) {perror("shmat");exit(1);}// 写入数据到共享内存strcpy(shmaddr, "Hello, Shared Memory!");printf("Data written to shared memory: %s\n", shmaddr);// 分离共享内存段if (shmdt(shmaddr) == -1) {perror("shmdt");exit(1);}// 删除共享内存段if (shmctl(shmid, IPC_RMID, 0) == -1) {perror("shmctl");exit(1);}return 0;
}

二、shm_open/shm_unlink函数

1、shm_open

  shm_open 是 POSIX 标准中定义的用于创建或打开共享内存对象的函数。与传统的 shmget 函数类似,shm_open 提供了一种不同的方式来管理共享内存,使用文件描述符来标识共享内存对象。shm_open 的原型如下:

int shm_open(const char *name, int oflag, mode_t mode);
  • name 参数是一个指向表示共享内存对象名称的字符串的指针。共享内存对象通过名称在文件系统中标识。
  • oflag 参数指定了打开共享内存对象时的操作标志,可以是一组位或运算的标志,比如 O_CREAT(创建共享内存对象), O_RDWR(可读写方式打开)等。
  • mode 参数指定了新创建的共享内存对象的权限标志,通常与 open 系统调用中使用的 mode 参数一样。

  使用 shm_open 函数创建或打开共享内存对象后,可以通过标准的文件 I/O 函数(如 readwrite)对共享内存进行读写操作。共享内存对象在文件系统中以名字为标识,不同进程可以通过相同的名字打开同一个共享内存对象,从而实现进程间通信。

  在使用 shm_open 函数时,需要注意以下几点:

  • 共享内存对象的名称必须在当前系统中是唯一的,以避免命名冲突。
  • 在打开共享内存对象后,应谨慎进行读写操作,避免发生竞争条件和数据一致性问题。
  • 当不再需要共享内存对象时,应当使用 shm_unlink 函数来释放该共享内存对象,以防止资源泄漏。

  总的来说,shm_open 提供了一种基于文件描述符的共享内存对象管理方式,使得共享内存在不同进程之间的使用更加灵活和方便。

2、shm_unlink

  shm_unlink 是一个函数,用于删除由 shm_open 创建的共享内存对象的名称。在 POSIX 标准中,共享内存对象在文件系统中以名称标识,使用 shm_open 打开该共享内存对象后,可以通过 shm_unlink 函数删除该对象的名字,但不会立即释放共享内存本身。只有在所有进程都关闭该共享内存对象的文件描述符或调用 shm_unlink 之后,共享内存对象才会被彻底删除。shm_unlink 的原型如下:

int shm_unlink(const char *name);
  • name 参数是一个指向表示要删除的共享内存对象名称的字符串的指针。

  使用 shm_unlink 函数可以确保在不再需要共享内存对象时清理对象的名称,这样可以避免其他进程再次通过该名称打开共享内存对象。但需要注意的是,即使调用了 shm_unlink,只有当所有打开该共享内存对象的文件描述符都关闭,共享内存对象才会被系统真正删除。

  一般来说,在不再需要使用共享内存对象时,建议及时调用 shm_unlink,以便及时释放已不需要的资源。这样可以更好地管理共享内存,避免资源泄漏和冗余共享内存对象的存在。

3、示例代码

  以下是一个简单的示例代码,展示了如何使用 shm_openshm_unlink 函数来创建和删除共享内存对象的名称。

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/mman.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>#define SHM_NAME "/my_shared_memory"
#define SHM_SIZE 1024int main() {const char *data = "Hello, this is shared memory demo!";// Create or open a shared memory objectint shm_fd = shm_open(SHM_NAME, O_CREAT | O_RDWR, S_IRUSR | S_IWUSR);ftruncate(shm_fd, SHM_SIZE);// Map the shared memory object into memorychar *shm_ptr = mmap(0, SHM_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, shm_fd, 0);// Write data to the shared memorystrcpy(shm_ptr, data);printf("Data written to shared memory: %s\n", shm_ptr);// Unmap the shared memorymunmap(shm_ptr, SHM_SIZE);// Close the shared memory objectclose(shm_fd);// Unlink the shared memory object nameshm_unlink(SHM_NAME);return 0;
}

  在这个示例代码中,首先使用 shm_open 函数创建或打开一个共享内存对象,并指定名称为 "/my_shared_memory"。然后使用 ftruncate 函数设置共享内存对象的大小为 SHM_SIZE 字节。

  接着,使用 mmap 函数将共享内存对象映射到当前进程的内存空间中,使得可以对其进行读写操作。将数据写入共享内存后,通过 munmap 函数解除共享内存的映射,并通过 close 函数关闭共享内存对象的文件描述符。

  最后,调用 shm_unlink 函数删除共享内存的名称,这样其他进程就无法再使用该名称打开共享内存对象。

三、课外阅读

推荐阅读:进程间通信–共享内存篇

  欢迎大家指导和交流!如果我有任何错误或遗漏,请立即指正,我愿意学习改进。期待与大家一起进步!

这篇关于Linux: shm_xx系列函数使用详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/766212

相关文章

Pandas使用SQLite3实战

《Pandas使用SQLite3实战》本文主要介绍了Pandas使用SQLite3实战,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录1 环境准备2 从 SQLite3VlfrWQzgt 读取数据到 DataFrame基础用法:读

JSON Web Token在登陆中的使用过程

《JSONWebToken在登陆中的使用过程》:本文主要介绍JSONWebToken在登陆中的使用过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录JWT 介绍微服务架构中的 JWT 使用结合微服务网关的 JWT 验证1. 用户登录,生成 JWT2. 自定义过滤

Linux卸载自带jdk并安装新jdk版本的图文教程

《Linux卸载自带jdk并安装新jdk版本的图文教程》在Linux系统中,有时需要卸载预装的OpenJDK并安装特定版本的JDK,例如JDK1.8,所以本文给大家详细介绍了Linux卸载自带jdk并... 目录Ⅰ、卸载自带jdkⅡ、安装新版jdkⅠ、卸载自带jdk1、输入命令查看旧jdkrpm -qa

Java中StopWatch的使用示例详解

《Java中StopWatch的使用示例详解》stopWatch是org.springframework.util包下的一个工具类,使用它可直观的输出代码执行耗时,以及执行时间百分比,这篇文章主要介绍... 目录stopWatch 是org.springframework.util 包下的一个工具类,使用它

Java进行文件格式校验的方案详解

《Java进行文件格式校验的方案详解》这篇文章主要为大家详细介绍了Java中进行文件格式校验的相关方案,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、背景异常现象原因排查用户的无心之过二、解决方案Magandroidic Number判断主流检测库对比Tika的使用区分zip

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

Java使用Curator进行ZooKeeper操作的详细教程

《Java使用Curator进行ZooKeeper操作的详细教程》ApacheCurator是一个基于ZooKeeper的Java客户端库,它极大地简化了使用ZooKeeper的开发工作,在分布式系统... 目录1、简述2、核心功能2.1 CuratorFramework2.2 Recipes3、示例实践3

springboot security使用jwt认证方式

《springbootsecurity使用jwt认证方式》:本文主要介绍springbootsecurity使用jwt认证方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录前言代码示例依赖定义mapper定义用户信息的实体beansecurity相关的类提供登录接口测试提供一

go中空接口的具体使用

《go中空接口的具体使用》空接口是一种特殊的接口类型,它不包含任何方法,本文主要介绍了go中空接口的具体使用,具有一定的参考价值,感兴趣的可以了解一下... 目录接口-空接口1. 什么是空接口?2. 如何使用空接口?第一,第二,第三,3. 空接口几个要注意的坑坑1:坑2:坑3:接口-空接口1. 什么是空接

Linux samba共享慢的原因及解决方案

《Linuxsamba共享慢的原因及解决方案》:本文主要介绍Linuxsamba共享慢的原因及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux samba共享慢原因及解决问题表现原因解决办法总结Linandroidux samba共享慢原因及解决