【Linux】18. 进程间通信 --- System V IPC(选学)

news2024/11/16 20:34:15

System V IPC

  • System V 消息队列
  • System V 共享内存
  • System V 信号量

system V 共享内存

在这里插入图片描述
共享内存区是最快的IPC形式。一旦这样的内存映射到共享它的进程的地址空间,这些进程间数据传递不再涉及到内核。
换句话说是进程不再通过执行进入内核的系统调用来传递彼此的数据

共享内存示意图

在这里插入图片描述

共享内存数据结构

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 */
 __kernel_time_t shm_dtime; /* last detach time */
 __kernel_time_t shm_ctime; /* last change time */
 __kernel_ipc_pid_t shm_cpid; /* pid of creator */
 __kernel_ipc_pid_t shm_lpid; /* pid of last operator */
 unsigned short shm_nattch; /* no. of current attaches */
 unsigned short shm_unused; /* compatibility */
 void *shm_unused2; /* ditto - used by DIPC */
 void *shm_unused3; /* unused */
};

共享内存函数

shmget函数

功能:用来创建共享内存
原型
int shmget(key_t key, size_t size, int shmflg);
参数
key:这个共享内存段名字
size:共享内存大小
shmflg:由九个权限标志构成,它们的用法和创建文件时使用的mode模式标志是一样的
返回值:成功返回一个非负整数,即该共享内存段的标识码;失败返回-1

在这里插入图片描述
深入理解key值:
OS需要对共享内存进行管理,既然要管理就遵循先描述再组织的原则
所以共享内存 = 内存块+共享内存的相关属性
共享内存的相关属性就是上述的struct shmid_ds数据结构进行管理,而key值就存储在shm_perm当中

shmat函数

功能:将共享内存段连接到进程地址空间
原型
void *shmat(int shmid, const void *shmaddr, int shmflg);
参数
shmid: 共享内存标识
shmaddr:指定连接的地址
shmflg:它的两个可能取值是SHM_RND和SHM_RDONLY
返回值:成功返回一个指针,指向共享内存第一个节;失败返回-1
说明
shmaddr为NULL,核心自动选择一个地址
shmaddr不为NULL且shmflg无SHM_RND标记,则以shmaddr为连接地址。
shmaddr不为NULL且shmflg设置了SHM_RND标记,则连接的地址会自动向下调整为SHMLBA的整数倍。公式:shmaddr -
(shmaddr % SHMLBA)
shmflg=SHM_RDONLY,表示连接操作用来只读共享内存

shmdt函数

功能:将共享内存段与当前进程脱离
原型
int shmdt(const void *shmaddr);
参数
shmaddr: 由shmat所返回的指针
返回值:成功返回0;失败返回-1
注意:将共享内存段与当前进程脱离不等于删除共享内存段

shmctl函数

功能:用于控制共享内存
原型
int shmctl(int shmid, int cmd, struct shmid_ds *buf);
参数
shmid:由shmget返回的共享内存标识码
cmd:将要采取的动作(有三个可取值)
buf:指向一个保存着共享内存的模式状态和访问权限的数据结构
返回值:成功返回0;失败返回-1

在这里插入图片描述

查看IPC资源命令:

在这里插入图片描述
在这里插入图片描述

代码实现

通过代码的方式进一步认清共享内存的使用

// comm.hpp 文件
#ifndef _COMM_HPP_
#define _COMM_HPP_

#include <iostream>
#include <cerrno>
#include <cstring>
#include <cstdlib>
#include <cstdio>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <unistd.h>

#define MAX_SIZE 4096

// 这里的PATHNAME为当前路径
#define PATHNAME "."
// 这里的PROJ_ID为随机值
#define PROJ_ID 0x66

key_t getKey()
{
    // ftok函数创建key值
    // ftok函数根据所提供的路径和值会通过算法确定一个唯一值
    // 当server和client看到同一个key值 也就意味着看到同一份共享内存
    key_t k = ftok(PATHNAME, PROJ_ID);
    if(k == -1)
    {
        std::cerr << errno << ": " << strerror(errno) << std::endl;
        exit(1);
    }
    return k;
}

int getShmHelper(key_t k, int flags)
{
    // shmget函数创建共享内存
    int shmid = shmget(k, MAX_SIZE, flags);
    if (shmid < 0)
    {
        std::cerr << errno << ": " << strerror(errno) << std::endl;
        exit(2);
    }
    return shmid;
}

int getshm(key_t k)
{
    return getShmHelper(k, IPC_CREAT);
}

int createShm(key_t k)
{
    // 创建新的shm权限为0600 只有自己有读写权限
    return getShmHelper(k, IPC_CREAT | IPC_EXCL | 0600);
}

#endif
// shm_server.cc 文件
#include "comm.hpp"

// server端进行创建和删除共享内存

int main()
{
    key_t k = getKey();
    printf("key:0x%x\n", k);
    int shmid = createShm(k);
    printf("shmid:%d\n", shmid);
    return 0;
}

在这里插入图片描述

// shm_client.cc文件
#include "comm.hpp"

int main()
{
    key_t k = getKey();
    printf("key:0x%x\n", k);
    int shmid = getshm(k);
    printf("shmid:%d\n", shmid);
    return 0;
} 

在这里插入图片描述

[hx@iZ0jl69kyvg0h181cozuf5Z shared_memory]$ ./shm_server 
key:0x66010470
17: File exists

出现 File exists 错误 说明文件已经存在
说明共享内存的生命周期是随OS的,而不是随进程的(不像管道,当没有文件描述符指向管道文件时,会自行关闭)

// comm.hpp文件
void *attachShm(int shmid)
{
    void *mem = shmat(shmid, nullptr, 0);
    // 这里为啥要强转成longlong类型呢?
    // 因为当前OS为64位 指针占8个字节
    if ((long long)mem == -1L)
    {
        // 挂接失败
        std::cerr << "shmat: " << errno << strerror(errno) << std::endl;
        exit(3);
    }
    return mem;
}

void detachShm(void *start)
{
    if (shmdt(start) == -1)
    {
        // 去关联失败
        std::cerr << "shmdt: " << errno << strerror(errno) << std::endl;
    }
}

void delShm(int shmid)
{
    if (shmctl(shmid, IPC_RMID, nullptr) == -1)
    {
        std::cerr << "shmctl: " << errno << strerror(errno) << std::endl;
    }
}
// shm_client.cc
#include "comm.hpp"

int main()
{
    key_t k = getKey();
    printf("key:0x%x\n", k);
    int shmid = getshm(k);
    printf("shmid:%d\n", shmid);

    sleep(5);

    char *start = (char*)attachShm(shmid);
    printf("attach success, address start: %p\n", start);

    sleep(5);

    detachShm(start);

    return 0;
} 
// shm_server.cc
#include "comm.hpp"

// server端进行创建和删除共享内存

int main()
{
    // 创建
    key_t k = getKey();
    printf("key:0x%x\n", k);
    int shmid = createShm(k);
    printf("shmid:%d\n", shmid);
    
    sleep(5);
    
    // 挂接
    // 这里为啥用start命名呢?
    // 因为挂接成功后返回的是进程地址空间的起始地址
    // 进一步加深先描述再组织的概念
    // OS不会直接让用户直接对共享内存进行操作,
    // 要先管理进进程地址空间中 
    char *start = (char*)attachShm(shmid);
    printf("attach success,address start:%p\n",start);

    sleep(5);

    // 去关联
    detachShm(start);

    sleep(10);

    // 删除共享内存
    delShm(shmid);
    return 0;
}

观察上述代码现象,如下所示:
在这里插入图片描述
进行最后一步通信:

// shm_server文件
    //通信
    while(true)
    {
        //直接从start中读取数据
        printf("client say:%s\n",start);
        sleep(1);
    }
// shm_client文件
    //通信
    const char* message = "hello server,我是另一个进程,正在与你进行通信";
    pid_t id = getpid();
    int cnt = 1;
    while(true)
    {
        sleep(1);
        // snprintf函数直接往start当中写入数据
        snprintf(start,MAX_SIZE,"%s[pid:%d][消息编号:%d]",message,id,cnt++);
    }

通信成功!!!
在这里插入图片描述
在这里插入图片描述
共享内存的缺点:
没有进行同步和互斥的操作,没有对数据做任何保护措施!
如果写入的很慢,读取的很快就总是会读取到重复冗余数据
在这里插入图片描述
进程独立性的原因:1. 各进程代码和数据独立 2.内核数据结构独立 3. 页表映射独立

system V 消息队列

在这里插入图片描述

system V 信号量

什么是信号量?

信号量的本质就是计数器,通常用来表示公共资源当中资源数量的多少

既然是计数器,那么能不能在代码中定义一个全局的count来进行统计呢?
答案:不能。如果是父子进程,往往会发生写时拷贝,二者看到的就不是同一份资源。(无法看到同一个count)
如果是毫不相干的进程,那么就更加需要提供通信技术来保证看到的是同一份资源

什么是公共资源?

公共资源就是指可以被多个进程同时进行访问的资源 (像:管道/共享内存/消息队列…)
而需要访问公共资源又会引发新的问题
在访问没有保护的公共资源就会出现数据不一致问题(类似于MySQL事务当中的脏读情况)
(假设现在的场景:输入abcd且abcd只有连在一起才有意义,但是输入一半就被另一个进程读取了)

为什么要让不同的进程看到同一份资源呢?

因为想要实现通信(进程间实现协同),但是进程具有独立性,如何让独立的进程实现协同呢? 让进程看到同一份资源
提出方法->引入了新的问题(数据不一致问题)

临界资源:未来被保护起来的公共资源
进程的大部分资源是独立的,只有少部分资源属于多进程共享,被多进程共享的资源称之为公共资源 而被保护起来的就叫做临界资源
资源(内存,文件,网络等)创建出来是要被使用的。

资源如何被进程使用呢?

一定是该进程有对应的代码来访问这部分临界资源: 临界区 非临界区

总结:
多进程在通信时本质是要看到一份公共的资源 ,这份公共资源未来被保护起来称其为临界资源,
访问临界资源的代码称之为临界区,不访问临界资源的代码称之为非临界区

保护资源的策略分为两种:互斥 && 同步(同步这里不涉及)

互斥: 当两个人同时访问一份资源时,只有当你访问完我才能访问
要么不做,要做就做完 只有两态的这种情况称之为原子性

共享资源的使用方式:1. 作为一个整体使用(管道/共享内存) 2. 划分为一个个的资源子部分使用(信号量)
对于共享资源划分成子部分(可以供不同的进程使用 – 并发) 而整体使用就是互斥串行(效率低)
在这里插入图片描述
所有的进程在访问公共资源之前,都必须先申请sem信号量 (先申请sem信号量的前提是所有进程必须先得看到同一个信号量)
信号量本身作为公共资源,信号量是不是也要保证自己的安全呢?
–,++信号量必须保证自身操作的安全性。
–,++操作是原子(只有两态:成功 or 失败)

申请信号量成功时相当于预定了共享资源当中的某一部分小资源,允许进程进行访问
如果申请信号量不成功,就不允许进程进入共享资源当中,进而达到保护共享资源以及其他进程的目的

通过计数器的方式,对临界资源进行保护 这种计数器我们称之为信号量

申请信号量后,进程可能会访问同一份子资源,需要程序员自己写代码保护资源

接口使用

在这里插入图片描述
在这里插入图片描述

system V 资源总结:

共享内存,消息队列,信号量,IPC资源 都是先描述再组织
操作系统为了维护IPC资源必须花费大量的数据结构来描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1656303.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Python图形界面(GUI)Tkinter笔记(一):根窗口的创建

Tkinter库是Python的内置关于图形界面编程&#xff08;GUI全称为Graphical User Interface&#xff0c;中文意思为“图形用户界面”&#xff09;的一个库。直接导入Tkinter使用即可。 其余笔记&#xff1a;【Python图形界面&#xff08;GUI&#xff09;Tkinter笔记&#xff08;…

ES6语法教程

简介&#xff1a; ECMA European Computer Manufactures Association 欧洲计算机制造商协会&#xff0c;该组织的目标是评估、开发、和认可电信和计算机标准&#xff0c;94年后该组织改名为Ecma国标。 ECMAScript是由Ecma国际通过ECMA-262标准化的脚本程序设计语言 Ecma国…

【设计模式】JAVA Design Patterns——Abstract-document

&#x1f50d; 目的 使用动态属性&#xff0c;并在保持类型安全的同时实现非类型化语言的灵活性。 &#x1f50d; 解释 抽象文档模式使您能够处理其他非静态属性。 此模式使用特征的概念来实现类型安全&#xff0c;并将不同类的属性分离为一组接口 真实世界例子 考虑由多个部…

九州金榜|如果孩子胆小懦弱,如何家庭教育?

在孩子成长的过程中&#xff0c;孩子可能会出现胆小懦弱的表现&#xff0c;当孩子出现这个问题是&#xff0c;势必会对孩子成长造成影响&#xff0c;可能会影响孩子的社交&#xff0c;学习以及日常生活等。对此很多家长不明白哪地方出现了问题&#xff0c;就会着急。面对这种情…

速度背!24上软考信管“经典100道母题来了”!

距离软考考试的时间越来越近了&#xff0c;趁着这两周赶紧准备起来 今天给大家整理了——信息系统管理工程师经典100道母题&#xff08;含解析&#xff09;&#xff0c;有PDF版&#xff0c;可打印&#xff0c;每天刷一点。 1、微机系统中&#xff0c;&#xff08; &#xff09…

一键自动化博客发布工具,用过的人都说好(cnblogs篇)

cnblogs和其他的博客平台相比会比较复杂&#xff0c;需要设置的项目也比较多一些&#xff0c;弄懂了cnblogs的实现方式&#xff0c;那么你应该对selenium的整个框架使用已经烂熟于心了。 除了正常的标题&#xff0c;内容&#xff0c;摘要之外&#xff0c;cnblogs还需要设置个人…

操作系统原理与实验——实验十三多道批处理作业调度(作业不可移动)

关键代码 #include<stdio.h> #include<stdlib.h> #include<string.h> typedef struct data{int hour;//当前小时int min;//当前分钟 }time; struct node{char name[20];//进程名time arrive;//到达就绪队列时间int zx;//执行时间(预期时间)int size;int ta…

用栈实现队列——leetcode刷题

题目要求我们只用栈的基本操作 push to top 入栈&#xff0c;peek from top 返回栈顶元素&#xff0c;pop from top 移除并返回栈顶元素&#xff0c;size 栈的大小&#xff0c;is_empty 判断栈是否为空&#xff0c;这几个函数来实现队列&#xff0c;也就是说&#xff0c;我们在…

建议收藏!如何选择高低温冲击试验箱厂家?

高低温冲击试验箱是实验室中不可或缺的重要设备之一&#xff0c;它在许多领域都发挥着关键作用。因此选择合适的高低温冲击试验箱厂家是非常关键的&#xff0c;因为正规厂家的正规产品能直接影响到产品实验的准确性和效率。下面我们一起来了解一下关于选择高低温冲击试验箱厂家…

从源头把控风险:集团多主体合规管理实战技巧分享

官.网地址&#xff1a;合合TextIn - 合合信息旗下OCR云服务产品 集团合规管理中&#xff0c;为了规避内外部利益冲突&#xff0c;需要对员工、供应商、经销商、客户、黑名单企业等多主体及其关联主体之间&#xff0c;进行多维度、多层级的关系挖掘与排查&#xff0c;避免利益…

OpenHarmony 实战开发(南向)-Docker编译环境搭建

Docker环境介绍 OpenHarmony为开发者提供了两种Docker环境&#xff0c;以帮助开发者快速完成复杂的开发环境准备工作。两种Docker环境及适用场景如下&#xff1a; 独立Docker环境&#xff1a;适用于直接基于Ubuntu、Windows操作系统平台进行版本编译的场景。 基于HPM的Docker…

react、vue 把要渲染的元素挂到指定节点。

react、vue 把要渲染的元素挂到指定节点。 文章目录 vue3 的 Teleportreact 的 createPortal 在日常工作中&#xff0c;需要封装一些定制的浮动模态框&#xff0c;但是每个弹框在不同的父级元素中&#xff0c;那弹框的层级我们是没有办法控制到的&#xff0c;所有有必要把它统一…

2024年钉钉直播回放怎么永久保存

钉钉群直播回放下载工具我已经打包好了&#xff0c; 钉钉群直播回放下载链接&#xff1a;百度网盘 请输入提取码 提取码&#xff1a;1234 --来自百度网盘超级会员V10的分享 1.首先解压好我给大家准备好的压缩包 2.再继续解压M3U8逍遥一仙下载器 3.进入M3U8逍遥一仙下载器文…

Pytorch快速上手

Pytorch快速上手 一、加载数据集 &#xff08;Dataset&#xff09; 加载数据集需要继承Dataset&#xff0c;通常情况下需要实现__init__方法、__getitem__方法以及__len__方法。 案例一&#xff1a; import osimport torch from torch.utils.data import Dataset from PIL …

SqlDeveloper字体大小设置

菜单栏字体&#xff1a; C:\Users\xxx\AppData\Roaming\SQL Developer\system19.2.1.247.2212\o.sqldeveloper 下的ide.properties 修改 Ide.FontSize.Aqua30 编辑器字体&#xff1a;

特定消谐技术:SHEPWM调制方法

简介 特定消谐技术是通过计算PWM波的开关时刻&#xff0c;消除特定的低阶谐波。其基本理论是恰当地控制逆变器脉宽调制电压的波形&#xff0c;通过脉宽平均法把逆变器输出的方波电压转换成等效的正弦波&#xff0c;以消除某些特定的谐波。本文对特定谐波消除方法的基本原理进行…

SpringBoot启动流程源码解析

目录 一、SpringApplication构造方法解析 1. web应用类型 2. BootstrapRegistryInitializer 3. ApplicationContextInitializer 4. ApplicationListener 5. 推断Main方法所在类 二、SpringApplication.run(String... args)方法解析 1.创建DefaultBootstrapContext 2.获…

算法day02

1、202. 快乐数 如上题所述&#xff1a; 在该题意规则下&#xff0c;所有的数字变化会有两种情况&#xff0c;其一最后是有的会变化成恒为1的数&#xff1b;其二是有的数会变化会呈现成有规律的环&#xff0c;分别如下图所示&#xff1a; 可以近似的理解为图一就是一个环&#…

Oracle 23ai rpm安装配置及问题处理

1.安装介质下载 Oracle 23ai 免费版本已经正式发布&#xff0c;Oracle官网提供免费的下载试用&#xff08;无需账号&#xff09;地址如下 官网下载和试用地址 Oracle Database 23ai Free&#xff1a; https://www.oracle.com/database/free/get-started 三种安装方式可选…

Cheetah3D for Mac - 轻松打造专业级3D作品

对于追求专业级3D作品的设计师来说&#xff0c;Cheetah3D for Mac无疑是一款不可多得的工具。 这款软件拥有强大的建模、渲染和动画功能&#xff0c;能够满足您在3D设计方面的各种需求。通过简单的操作&#xff0c;您可以轻松构建出复杂的3D模型&#xff0c;并为其添加逼真的材…