Linux——共享内存

目录

一、共享内存概念

二、共享内存的一些函数

[2.1 shmget 创建共享内存](#2.1 shmget 创建共享内存)

[2.2 shmat 访问共享内存](#2.2 shmat 访问共享内存)

[2.3 shmdt 解除共享内存的映射](#2.3 shmdt 解除共享内存的映射)

[2.4 shnctl 删除共享内存段](#2.4 shnctl 删除共享内存段)

三、共享内存

[3.1 创建测试进程](#3.1 创建测试进程)

[3.2 使用循环测试](#3.2 使用循环测试)

​编辑

[3.3 共享内存写入程序](#3.3 共享内存写入程序)

[3.4 带有信号量的共享内存](#3.4 带有信号量的共享内存)


一、共享内存概念

共享内存是一种进程间通信的方式,允许多个进程访问和操作同一块内存区域。这样的内存区域被所有共享它的进程所拥有,进程可以将数据写入共享内存区域,也可以从中读取数据。共享内存在提高进程通信效率和降低开销方面具有优势,但也需要进行同步和互斥操作以避免数据竞争和冲突。

共享内存通常适用于需要高效地在进程间传递大量数据的场景,比如多个进程需要共享大型数据结构、图形图像处理或多媒体应用等。在使用共享内存时,需要注意管理内存的权限、同步访问和处理异常情况等问题,以确保数据的一致性和安全性。

二、共享内存的一些函数

2.1 shmget 创建共享内存

它的函数原型如下:

cpp 复制代码
int shmget(key_t key, size_t size, int shmflg);

参数含义:

  1. key: 用于标识共享内存段的键值,不同的进程使用相同的 key 值可以获取到同一个共享内存
  2. size: 创建共享内存时,指定要申请的共享内存空间大小
  3. shmflg: 用于指定创建共享内存段的访问权限和其他标志,比如权限位和内存段的创建方式等。常见的标志包括IPC_CREAT(如果不存在则创建共享内存段)和 IPC_EXCL(如果存在则返回错误)。
    shmget() 成功则返回共享内存的 ID, 失败返回-1
cpp 复制代码
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/shm.h>

int main()
{
    int shmid = shmget((key_t)2345,128,IPC_CREAT|0600);
    if(shmid == -1)
    {
        exit(1);
    }

}

2.2 shmat 访问共享内存

函数**shmat()**用于将共享内存段附加到调用进程的地址空间,并返回一个指向共享内存段起始地址的指针。

它的函数原型如下:

cpp 复制代码
void* shmat(int shmid, const void *shmaddr, int shmflg);

参数含义:

  1. shmid: 表示要附加的共享内存段的标识符,通常是由shmget()函数返回的共享内存标识符。

  2. shmaddr: 指定共享内存段连接到调用进程地址空间的地址。一般设置为 NULL,由系统自动选择映射的虚拟地址空间

  3. shmflg: 用以指定附加共享内存段的附加方式。常见的标志包括SHM_RDONLY(只读方式附加共享内存段)和 SHM_RND(指示共享内存段将在系统限定的地址范围内附加)。 一般给 0, 可以给 SHM_RDONLY 为只读模式,其他的为读写

返回值:
函数的返回值是一个指针,指向共享内存段的起始地址。 shmat()成功则返回共享内存的首地址,失败返回 NULL

cpp 复制代码
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/shm.h>

int main()
{
    int shmid = shmget((key_t)2345,128,IPC_CREAT|0600);
    if(shmid == -1)
    {
        exit(1);
    }

    char* s=(char*)shmat(shmid,NULL,0);
    if(s==(char*)-1)
    {
        exit(1);
    }

    strcpy(s,"hello");

    shmdt(s);

}

2.3 shmdt 解除共享内存的映射

当进程不再需要访问共享内存时,需要使用 shmdt 函数将共享内存段从进程地址空间中解除映射。其函数原型如下:

cpp 复制代码
int shmdt(const void *shmaddr);

参数含义:

  1. shmaddr: 表示要分离的共享内存段的起始地址。通常是由shmat()函数返回的指针,指向共享内存段的起始位置。

返回值:

当成功分离共享内存段时,函数返回0。如果出现错误,函数会返回-1,并且可以通过检查errno变量来获取错误信息。

2.4 shnctl 删除共享内存段

函数**shmctl()**用于控制共享内存段的属性

cpp 复制代码
int shmctl(int shmid, int cmd, struct shmid_ds *buf);

参数含义:

  1. shmid: 共享内存段的标识符,通常由shmget()函数返回。

  2. cmd: 控制命令,指定对共享内存段执行的操作类型。常见的命令有:

    • IPC_STAT: 获取共享内存段的状态信息,并将其存储在buf指向的shmid_ds结构体中。
    • IPC_SET: 设置共享内存段的权限模式,需要提供buf指向的shmid_ds结构体。
    • IPC_RMID: 从系统中删除共享内存段,同时释放其占用的资源。
  3. buf: 指向一个shmid_ds结构体的指针,用于存储或传递共享内存段的状态信息或权限设置。

返回值:

函数成功执行时返回0,否则返回-1,并通过设置errno变量来指示错误类型。

三、共享内存

3.1 创建测试进程

cpp 复制代码
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/shm.h>

int main()
{
    int shmid = shmget((key_t)2345,128,IPC_CREAT|0600);
    if(shmid == -1)
    {
        exit(1);
    }

    char* s=(char*)shmat(shmid,NULL,0);
    if(s==(char*)-1)
    {
        exit(1);
    }

    printf("s=%s\n",s);// 打印共享内存的地址
    shmdt(s);

}

运行结果:

3.2 使用循环测试

cpp 复制代码
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/shm.h>

int main()
{
    int shmid = shmget((key_t)2345,128,IPC_CREAT|0600);
    if(shmid == -1)
    {
        exit(1);
    }

    char* s=(char*)shmat(shmid,NULL,0);
    if(s==(char*)-1)
    {
        exit(1);
    }

    while(1)
    {
        if(strncmp(s,"end",3) == 0)
        {
            break;
        }
        printf("s=%s\n",s);
        sleep(1);
    }
    shmdt(s);

}

3.3 共享内存写入程序

cpp 复制代码
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/shm.h>

int main()
{
    int shmid = shmget((key_t)2345,128,IPC_CREAT|0600);
    if(shmid == -1)
    {
        exit(1);
    }

    char* s=(char*)shmat(shmid,NULL,0);
    if(s==(char*)-1)
    {
        exit(1);
    }

    while(1)
    {
        printf("input\n");
        char buff[128]={0};
        fgets(buff,128,stdin);

        strcpy(s,buff);
        if( strncmp(buff,"end",3) == 0)
        {
            break;
        }
    }
    shmdt(s);
}

3.4 带有信号量的共享内存

首先要创建两个信号量,一个设为0,另一个设为1。

头文件代码sem.h

cpp 复制代码
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/sem.h>

enum INDEX{SEM1=0,SEM2};

union semun
{
    int val;
};

void sem_init();//创建并初始化信号量
void sem_p(int index);
void sem_v(int index);
void sem_destroy();

sem.c代码

cpp 复制代码
#include"sem.h"

static int semid=-1;


void sem_init()//创建并初始化信号量
{
    semid=semget((key_t)1234,2,IPC_CREAT|IPC_EXCL|0600);
    if(semid==-1)
    {
        semid=semget((key_t)1234,2,0600);
        if(semid==-1)
        {
            printf("semget err\n");
        }
    }
    else
    {
        int arr[2]={1,0};
        for(int i=0;i<2;i++)
        {
            union semun a;
            a.val=arr[i];
            if(semctl(semid,i,SETVAL,a)==-1)
            {
                printf("semctl init err\n");
            }
        }
    }
}

void sem_p(int index)
{
    struct sembuf buf;
    buf.sem_num=index;
    buf.sem_op=-1;//p
    buf.sem_flg=SEM_UNDO;

    if(semop(semid,&buf,1) == -1)
    {
        printf("op p err\n");
    }
}

void sem_v(int index)
{
    struct sembuf buf;
    buf.sem_num=index;
    buf.sem_op=1;//v
    buf.sem_flg=SEM_UNDO;

    if(semop(semid,&buf,1) == -1)
    {
        printf("op v err\n");
    }
}

void sem_destroy()
{
    if(semctl(semid,0,IPC_RMID) == -1)
    {
        printf("semctl del\n");
    }
}

使用信号量的main.c代码

cpp 复制代码
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/shm.h>
#include"sem.h"

int main()
{
    int shmid = shmget((key_t)2345,128,IPC_CREAT|0600);
    if(shmid == -1)
    {
        exit(1);
    }

    char* s=(char*)shmat(shmid,NULL,0);
    if(s==(char*)-1)
    {
        exit(1);
    }

    sem_init();//创建 初始化
    while(1)
    {
        printf("input\n");
        char buff[128]={0};
        fgets(buff,128,stdin);

        sem_p(SEM1);//p  s1
        strcpy(s,buff);
        sem_v(SEM2);//v  s2
        if( strncmp(buff,"end",3) == 0)
        {
            break;
        }
    }
    shmdt(s);
}

测试代码test.c

cpp 复制代码
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#include<sys/shm.h>
#include"sem.h"

int main()
{
    int shmid = shmget((key_t)2345,128,IPC_CREAT|0600);
    if(shmid == -1)
    {
        exit(1);
    }

    char* s=(char*)shmat(shmid,NULL,0);
    if(s==(char*)-1)
    {
        exit(1);
    }

    sem_init();
    while(1)
    {
        sem_p(SEM2);
        if(strncmp(s,"end",3) == 0)
        {
            break;
        }
        
        printf("s=%s\n",s);
        sem_v(SEM1);
    }
    shmdt(s);
    sem_destroy();
    exit(0);
}

运行结果:

相关推荐
撸码到无法自拔1 小时前
云计算-私有云-私有云服务运维
运维·云计算
云边有个稻草人1 小时前
【Linux系统】第四节—详解yum+vim
linux·vim·yum·软件包管理器·linux软件生态·linux编辑器-vim使⽤·yum具体操作
小陶来咯1 小时前
【高级IO】多路转接之单线程Reactor
服务器·网络·数据库·c++
极小狐3 小时前
如何使用极狐GitLab 软件包仓库功能托管 maven?
java·运维·数据库·安全·c#·gitlab·maven
dz小伟4 小时前
vim的配置
linux·编辑器·vim
檀越剑指大厂5 小时前
【Docker系列】docker inspect查看容器部署位置
运维·docker·容器
江湖人称-杰6 小时前
CentOS配置了镜像源之后依旧下载元数据失败
linux·运维·centos
阿运河7 小时前
如何配置 VScode 断点调试Linux 工程代码
linux·ide·vscode
BXCQ_xuan7 小时前
DNS负载均衡和CDN的区别
运维·负载均衡
Xena_Networks8 小时前
SierraNet协议分析使用指导[RDMA]| 如何设置 NVMe QP 端口以进行正确解码
linux·服务器·网络