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

2024-05-10 06:12

本文主要是介绍【Linux】18. 进程间通信 --- System V IPC(选学),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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 0x66key_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资源必须花费大量的数据结构来描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

这篇关于【Linux】18. 进程间通信 --- System V IPC(选学)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Linux换行符的使用方法详解

《Linux换行符的使用方法详解》本文介绍了Linux中常用的换行符LF及其在文件中的表示,展示了如何使用sed命令替换换行符,并列举了与换行符处理相关的Linux命令,通过代码讲解的非常详细,需要的... 目录简介检测文件中的换行符使用 cat -A 查看换行符使用 od -c 检查字符换行符格式转换将

Linux系统配置NAT网络模式的详细步骤(附图文)

《Linux系统配置NAT网络模式的详细步骤(附图文)》本文详细指导如何在VMware环境下配置NAT网络模式,包括设置主机和虚拟机的IP地址、网关,以及针对Linux和Windows系统的具体步骤,... 目录一、配置NAT网络模式二、设置虚拟机交换机网关2.1 打开虚拟机2.2 管理员授权2.3 设置子

Linux系统中卸载与安装JDK的详细教程

《Linux系统中卸载与安装JDK的详细教程》本文详细介绍了如何在Linux系统中通过Xshell和Xftp工具连接与传输文件,然后进行JDK的安装与卸载,安装步骤包括连接Linux、传输JDK安装包... 目录1、卸载1.1 linux删除自带的JDK1.2 Linux上卸载自己安装的JDK2、安装2.1

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

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

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

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

新特性抢先看! Ubuntu 25.04 Beta 发布:Linux 6.14 内核

《新特性抢先看!Ubuntu25.04Beta发布:Linux6.14内核》Canonical公司近日发布了Ubuntu25.04Beta版,这一版本被赋予了一个活泼的代号——“Plu... Canonical 昨日(3 月 27 日)放出了 Beta 版 Ubuntu 25.04 系统镜像,代号“Pluc

Linux安装MySQL的教程

《Linux安装MySQL的教程》:本文主要介绍Linux安装MySQL的教程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux安装mysql1.Mysql官网2.我的存放路径3.解压mysql文件到当前目录4.重命名一下5.创建mysql用户组和用户并修

Linux上设置Ollama服务配置(常用环境变量)

《Linux上设置Ollama服务配置(常用环境变量)》本文主要介绍了Linux上设置Ollama服务配置(常用环境变量),Ollama提供了多种环境变量供配置,如调试模式、模型目录等,下面就来介绍一... 目录在 linux 上设置环境变量配置 OllamPOgxSRJfa手动安装安装特定版本查看日志在

Linux系统之主机网络配置方式

《Linux系统之主机网络配置方式》:本文主要介绍Linux系统之主机网络配置方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、查看主机的网络参数1、查看主机名2、查看IP地址3、查看网关4、查看DNS二、配置网卡1、修改网卡配置文件2、nmcli工具【通用

Linux系统之dns域名解析全过程

《Linux系统之dns域名解析全过程》:本文主要介绍Linux系统之dns域名解析全过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、dns域名解析介绍1、DNS核心概念1.1 区域 zone1.2 记录 record二、DNS服务的配置1、正向解析的配置