Linux进程间通信(System V) --- 共享内存

news/2024/7/20 14:36:47 标签: 数据结构与算法, 内存管理
共享内存 IPC 原理

共享内存进程间通信机制主要用于实现进程间大量的数据传输,下图所示为进程间使用共享内存实现大量数据传输的示意图:

1319058-20180207212445857-1940174400.png

共享内存是在内存中单独开辟的一段内存空间,这段内存空间有自己特有的数据结构,包括访问权限、大小和最近访问的时间等。该数据结构定义如下:

from /usr/include/linux/shm.h

struct shmid_ds {
    struct ipc_perm     shm_perm;   /* operation perms 操作权限 */
    int         shm_segsz;  /* size of segment (bytes) 段长度大小 */
    __kernel_time_t     shm_atime;  /* last attach time 最近attach时间 */
    __kernel_time_t     shm_dtime;  /* last detach time 最近detach时间 */
    __kernel_time_t     shm_ctime;  /* last change time 最近change时间 */
    __kernel_ipc_pid_t  shm_cpid;   /* pid of creator 创建者pid */
    __kernel_ipc_pid_t  shm_lpid;   /* pid of last operator 最近操作pid */
    unsigned short      shm_nattch; /* no. of current attaches */
    unsigned short      shm_unused; /* compatibility */
    void            *shm_unused2;   /* ditto - used by DIPC */
    void            *shm_unused3;   /* unused */
};

两个进程在使用此共享内存空间之前,需要在进程地址空间与共享内存空间之间建立联系,即将共享内存空间挂载到进程中。

系统对共享内存做了以下限制:

#define SHMMAX 0x2000000         /* max shared seg size (bytes) 最大共享段大小 */
#define SHMMIN 1             /* min shared seg size (bytes) 最小共享段大小 */
#define SHMMNI 4096          /* max num of segs system wide */
#define SHMALL (SHMMAX/getpagesize()*(SHMMNI/16))
#define SHMSEG SHMMNI            /* max shared segs per process */
内存管理">Linux 共享内存管理

1.创建共享内存

#include <sys/ipc.h>
#include <sys/shm.h>

/*
 * 第一个参数为 key 值,一般由 ftok() 函数产生
 * 第二个参数为欲创建的共享内存段大小(单位为字节)
 * 第三个参数用来标识共享内存段的创建标识
 */
int shmget(key_t key, size_t size, int shmflg);

2.共享内存控制

#include <sys/ipc.h>
#include <sys/shm.h>

/*
 * 第一个参数为要操作的共享内存标识符
 * 第二个参数为要执行的操作
 * 第三个参数为 shmid_ds 结构的临时共享内存变量信息
 */
int shmctl(int shmid, int cmd, struct shmid_ds *buf);

3.映射共享内存对象

系统调用 shmat() 函数实现将一个共享内存段映射到调用进程的数据段中,并返回内存空间首地址,其函数声明如下:

#include <sys/types.h>
#include <sys/shm.h>

/*
 * 第一个参数为要操作的共享内存标识符
 * 第二个参数用来指定共享内存的映射地址,非0则为此参数,为0的话由系统分配
 * 第三个参数用来指定共享内存段的访问权限和映射条件
 */
void *shmat(int shmid, const void *shmaddr, int shmflg);

4.分离共享内存对象

在使用完毕共享内存空间后,需要使用 shmdt() 函数调用将其与当前进程分离。函数声明如下:

#include <sys/types.h>
#include <sys/shm.h>

/*
 * 参数为分配的共享内存首地址
 */
int shmdt(const void *shmaddr);
共享内存在父子进程间遵循的约定
1.使用 fork() 函数创建一个子进程后,该进程继承父亲进程挂载的共享内存。
2.如果调用 exec() 执行一个新的程序,则所有挂载的共享内存将被自动卸载。
3.如果在某个进程中调用了 exit() 函数,所有挂载的共享内存将与当前进程脱离关系。
程序实例

申请一段共享内存,父进程在首地址处存入一整数,子进程读出。

#include <stdio.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdlib.h>

#define SHM_SIZE 1024

int main()
{
    int shm_id, pid;
    int *ptr = NULL;

    /* 申请共享内存 */
    shm_id = shmget((key_t)1004, SHM_SIZE, IPC_CREAT | 0600);

    /* 映射共享内存到进程地址空间 */
    ptr = (int*)shmat(shm_id, 0, 0);

    printf("Attach addr is %p \n", ptr);

    *ptr = 1004;

    printf("The Value of Parent is : %d \n", *ptr);

    if((pid=fork()) == -1){
        perror("fork Err");
        exit(0);
    }
    else if(!pid){
        printf("The Value of Child is : %d \n", *ptr);
        exit(0);
    }else{
        sleep(1);

        /* 解除映射 */
        shmdt(ptr);
        
        /* 删除共享内存 */
        shmctl(shm_id, IPC_RMID, 0);
    }

    return 0;
}

输出结果:

1319058-20180207212433513-212715131.png

转载于:https://www.cnblogs.com/GyForever1004/p/8428506.html


http://www.niftyadmin.cn/n/1441916.html

相关文章

VB+Oracle 9i如何构建开发和运行的环境(转)

VBOracle 9i如何构建开发和运行的环境&#xff1a; 如何选择前台开发工具如何构建开发和运行环境 一个VBOracle 9i的数据库应用从逻辑上看有3个组成部分。如图10.1所示。 1. 构建开发和运行客户机 在开发客户机上要进行以下工作。 &#xff08;1&#xff09;安装VB。 &#xf…

Day_07 传智健康项目-Freemarker

传智健康项目 第7章 1. 页面静态化介绍 本章课程中我们已经实现了移动端套餐列表页面和套餐详情页面的动态展示。但是我们需要思考一个问题&#xff0c;就是对于这两个页面来说&#xff0c;每次用户访问这两个页面都需要查询数据库获取动态数据进行展示&#xff0c;而且这两个…

安全响应中心 — 垃圾邮件事件报告(6.28)

2023年6月 第四周 样本概况 ✅ 类型1&#xff1a;伪造正常转发邮件&#xff08;链接&#xff09; 钓鱼邮件一直是邮件防护绕不开的话题。近日安全团队收到了一批钓鱼邮件&#xff0c;发送者将正常邮件内容和钓鱼内容拼凑在一起&#xff0c;将一封钓鱼邮件伪装成经过转发的正…

SmartPhone 2003 手机编程实战之一:简单上手(转)

SmartPhone 2003 手机编程实战之一&#xff1a;简单上手今天买了一款 多普达565的智能手机&#xff0c;加上了128M的MiniSD卡&#xff0c;基本可以满足编程的需求了&#xff0c;便随手写了一个最简单的HelloWord&#xff0c;并到手机上调试(部署到菜单中暂时没有实现)。一、基本…

Leetcode 012 Integer to Roman(模拟)

题目连接&#xff1a;Leetcode 012 Integer to Roman 解题思路&#xff1a;模拟。 class Solution {public:string intToRoman(int num) {string ans;int p[] {1000, 900, 500, 400, 100, 90, 50, 40, 10, 9, 5, 4, 1};string s[] {"M", "CM", "D&q…

导致 KEIL error #20 的一种情况

> 描述 <> 环境为 KEIL5.20 & STM32F429工程&#xff0c;平台为 win10 <> 结构体原形如下 File <A.h> #include "all.h" 1 /* Define the GPS structure ---------------------------------------------------------------------*/2 typede…

CDMA无线数据业务发展状况和发展前景(转)

今年的3月28日&#xff0c;中国联通在钓鱼台国宾馆开了发布会&#xff0c;正式宣布互动世界、彩E和掌中宽带三项业务正式试运行。7月1日&#xff0c;这三项业务正式投入运营。7月22日联通又开通了定位之星和神奇宝典业务&#xff0c;加上去年就开通的联通在信业务&#xff0c;联…

JVM--------3

一、为什么要使用类加载器&#xff1f;Java语言里&#xff0c;类加载都是在程序运行期间完成的&#xff0c;这种策略虽然会令类加载时稍微增加一些性能开销&#xff0c;但是会给java应用程序提供高度的灵活性。例如&#xff1a;1.编写一个面向接口的应用程序&#xff0c;可能等…