进程间通信——共享内存

原文来源:(4条消息) 【转】进程间的五种通信方式_进程间的5种通信方式_哈市雪花的博客-CSDN博客icon-default.png?t=N176https://blog.csdn.net/baidu_38621657/article/details/105724822?ops_request_misc=%257B%2522request%255Fid%2522%253A%2522167860274416800222885633%2522%252C%2522scm%2522%253A%252220140713.130102334..%2522%257D&request_id=167860274416800222885633&biz_id=0&utm_medium=distribute.pc_search_result.none-task-blog-2~all~top_positive~default-1-105724822-null-null.142^v73^control,201^v4^add_ask,239^v2^insert_chatgpt&utm_term=%E8%BF%9B%E7%A8%8B%E9%97%B4%E7%9A%84%E4%BA%94%E7%A7%8D%E9%80%9A%E4%BF%A1%E6%96%B9%E5%BC%8F&spm=1018.2226.3001.4187

共享内存(Shared Memory),指两个或多个进程共享一个给定的存储区。

1、特点

  1. 共享内存是最快的一种 IPC,因为进程是直接对内存进行存取。

  2. 因为多个进程可以同时操作,所以需要进行同步。

  3. 信号量+共享内存通常结合在一起使用,信号量用来同步对共享内存的访问。

2、原型

1 #include <sys/shm.h>
2 // 创建或获取一个共享内存:成功返回共享内存ID,失败返回-1
3 int shmget(key_t key, size_t size, int flag);
4 // 连接共享内存到当前进程的地址空间:成功返回指向共享内存的指针,失败返回-1
5 void *shmat(int shm_id, const void *addr, int flag);
6 // 断开与共享内存的连接:成功返回0,失败返回-1
7 int shmdt(void *addr); 
8 // 控制共享内存的相关信息:成功返回0,失败返回-1
9 int shmctl(int shm_id, int cmd, struct shmid_ds *buf);

当用shmget函数创建一段共享内存时,必须指定其 size;而如果引用一个已存在的共享内存,则将 size 指定为0 。

当一段共享内存被创建以后,它并不能被任何进程访问。必须使用shmat函数连接该共享内存到当前进程的地址空间,连接成功后把共享内存区对象映射到调用进程的地址空间,随后可像本地空间一样访问。

shmdt函数是用来断开shmat建立的连接的。注意,这并不是从系统中删除该共享内存,只是当前进程不能再访问该共享内存而已。

shmctl函数可以对共享内存执行多种操作,根据参数 cmd 执行相应的操作。常用的是IPC_RMID(从系统中删除该共享内存)。

3、例子

下面这个例子,使用了【共享内存+信号量+消息队列】的组合来实现服务器进程与客户进程间的通信。

  • 共享内存用来传递数据;
  • 信号量用来同步;
  • 消息队列用来 在客户端修改了共享内存后 通知服务器读取。

shm_write.c

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


//int shmget(key_t key, size_t size, int shmflg);

/*
2 // 创建或获取一个共享内存:成功返回共享内存ID,失败返回-1
3 int shmget(key_t key, size_t size, int flag);
4 // 连接共享内存到当前进程的地址空间:成功返回指向共享内存的指针,失败返回-1
5 void *shmat(int shm_id, const void *addr, int flag);
6 // 断开与共享内存的连接:成功返回0,失败返回-1
7 int shmdt(void *addr); 
8 // 控制共享内存的相关信息:成功返回0,失败返回-1
9 int shmctl(int shm_id, int cmd, struct shmid_ds *buf);
*/


int main()
{
        int shmid;
        char *shmaddr;

        key_t key;
        key = ftok(".",1);

        // 创建或获取一个共享内存:成功返回共享内存ID,失败返回-1
        //int shmget(key_t key, size_t size, int flag);
        shmid = shmget(key,1024*4,IPC_CREAT|0666);
        if(shmid==-1){
                printf("shmget no ok\n");
                exit(-1);
        }

        // 连接共享内存到当前进程的地址空间:成功返回指向共享内存的指针,失败返回-1
        // void *shmat(int shm_id, const void *addr, int flag);
        shmaddr =  shmat(shmid,0,0);

        printf("shmat ok\n");
        strcpy(shmaddr,"hello world");

        sleep(5);

        // 断开与共享内存的连接:成功返回0,失败返回-1
        //int shmdt(void *addr);
        shmdt(shmaddr);

        // 控制共享内存的相关信息:成功返回0,失败返回-1
        // int shmctl(int shm_id, int cmd, struct shmid_ds *buf);
        shmctl(shmid,IPC_RMID,0);

        printf("quit\n");

        return 0;
}

shm_read,cyunxinyu

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


//int shmget(key_t key, size_t size, int shmflg);

/*
2 // 创建或获取一个共享内存:成功返回共享内存ID,失败返回-1
3 int shmget(key_t key, size_t size, int flag);
4 // 连接共享内存到当前进程的地址空间:成功返回指向共享内存的指针,失败返回-1
5 void *shmat(int shm_id, const void *addr, int flag);
6 // 断开与共享内存的连接:成功返回0,失败返回-1
7 int shmdt(void *addr); 
8 // 控制共享内存的相关信息:成功返回0,失败返回-1
9 int shmctl(int shm_id, int cmd, struct shmid_ds *buf);
*/


int main()
{
        int shmid;
        char *shmaddr;

        key_t key;
        key = ftok(".",1);

        // 创建或获取一个共享内存:成功返回共享内存ID,失败返回-1
        //int shmget(key_t key, size_t size, int flag);
        shmid = shmget(key,1024*4,0);
        if(shmid==-1){
                printf("shmget no ok\n");
                exit(-1);
        }

        // 连接共享内存到当前进程的地址空间:成功返回指向共享内存的指针,失败返回-1
        // void *shmat(int shm_id, const void *addr, int flag);
        shmaddr =  shmat(shmid,0,0);

        printf("shmat ok\n");
        printf("data : %s\n",shmaddr);



        // 断开与共享内存的连接:成功返回0,失败返回-1
        //int shmdt(void *addr);
        shmdt(shmaddr);


        printf("quit\n");

        return 0;
}

运行效果

 

 

另附: 

命令:ipcs -m  查看有哪些共享内存

        ipcrm -m id号  删除共享内存

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值