IO进程线程(十二)进程间通信 共享内存 信号灯集

2024-06-10 19:20

本文主要是介绍IO进程线程(十二)进程间通信 共享内存 信号灯集,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 一、共享内存 shared memory(shm)
    • (一)特点
    • (二) 相关API
      • 1. 创建共享内存
      • 2. 映射共享内存到当前的进程空间
      • 3. 取消地址映射
      • 4. 共享内存控制
    • (三)使用示例
    • (四) 属性
  • 二、信号灯集---控制进程间同步
    • (一)特点
    • (二) 相关API
      • 1. 创建一个信号灯集
      • 2. 信号灯集控制函数
      • 3. 信号灯集的操作函数
    • (三)封装函数

一、共享内存 shared memory(shm)

(一)特点

在内核中创建共享内存,让进程A和进程B都能够访问到,通过这段内存进行数据的传递。
共享内存是所有进程间通信方式中效率最高的(不需要来回进行数据的拷贝)
在这里插入图片描述

(二) 相关API

1. 创建共享内存

#include <sys/ipc.h>
#include <sys/shm.h>int shmget(key_t key, size_t size, int shmflg);
功能:创建共享内存
参数:key:键值key 通过ftok获取IPC_PRIVATE:只能用于亲缘进程间的通信size:共享内存的大小  PAGE_SIZE(4k)的整数倍shmflg:共享的标志位IPC_CREAT|0666 或 IPC_CREAT|IPC_EXCL|0666
返回值:成功 共享内存编号失败 -1 重置错误码
  • 注:
  • 共享内存大小必须要4k的整数倍,因为一页是4k。如果申请时不要求4
    的整数倍,分配时也是分配4k的整数倍。
  • 内核空间越界会直接段错误
  • 同一个key值可以同时用于消息队列,共享内存,信号灯集

2. 映射共享内存到当前的进程空间

#include <sys/ipc.h>
#include <sys/shm.h>
void *shmat(int shmid, const void *shmaddr, int shmflg);
功能:映射共享内存到当前的进程空间
参数:shmid:共享内存编号shmaddr:指定共享内存出现在进程内存地址的什么位置,直接指定为NULL,让系统自动分配shmflg:共享内存操作方式0    读写SHM_RDONLY    只读
返回值:成功 指向共享内存的地址失败 (void *)-1 重置错误码

3. 取消地址映射

#include <sys/ipc.h>
#include <sys/shm.h>
int shmdt(const void *shmaddr);
功能:取消地址映射
参数:shmaddr:指向共享内存的指针
返回值:成功 0失败 -1 重置错误码

4. 共享内存控制

int shmctl(int shmid, int cmd, struct shmid_ds *buf);
功能:共享内存控制的函数
参数:shmid:共享内存编号cmd:操作的命令码IPC_STAT:获取IPC_SET:设置IPC_RMID:删除共享内存标记要销毁的段。实际上,只有在最后一个进程将其分离之后 (也就是说,关联结构shmid_ds的shm_nattch成员为零时), 段才会被销毁。调用者必须是段的所有者或创建者,或具有特权。buf参数被忽略。buf:共享内存属性结构体指针
返回值:成功 0失败 -1 重置错误码

(三)使用示例

read.c

#include <my_head.h>#define SHM_PAGE 1024*4
int main(int argc, char const *argv[])
{//获取键值key_t key = ftok("/home/linux/05work",'A');if(-1 == key)ERR_LOG("ftok error");//创建共享内存int shmid = shmget(key, SHM_PAGE,IPC_CREAT|0666);if(-1 == shmid)ERR_LOG("shmget error");//映射共享内存char *addr = (char *)shmat(shmid,NULL,SHM_RDONLY);while(1){getchar();//防止刷屏,回车一次打印一次  if(!strcmp(addr,"quit")){break;}    printf("%s",addr);}//解除映射if(-1 == shmdt(addr)){ERR_LOG("shmid error");}//第三个参数会被忽略if(-1 == shmctl(shmid,IPC_RMID,NULL)){if(EINVAL == errno){return 0;}ERR_LOG("shmctl error");}return 0;
}

write.c

#include <my_head.h>#define SHM_PAGE 1024*4
int main(int argc, char const *argv[])
{//获取键值key_t key = ftok("/home/linux/05work",'A');if(-1 == key)ERR_LOG("ftok error");//创建共享内存int shmid = shmget(key, SHM_PAGE,IPC_CREAT|0666);if(-1 == shmid)ERR_LOG("shmget error");//映射共享内存char *addr = (char *)shmat(shmid,NULL,0);while(1){printf("请输入要发送的内容:");scanf("%s",addr);if(!strcmp(addr,"quit")){break;}}//解除映射if(-1 == shmdt(addr)){ERR_LOG("shmid error");}//销毁if(-1 == shmctl(shmid,IPC_RMID,NULL)){if(EINVAL == errno){return 0;}ERR_LOG("shmctl error");}return 0;
}

(四) 属性

struct shmid_ds{struct ipc_perm shm_perm;    //权限结构体size_t shm_segsz;             //共享内存大小,单位是字节 __time_t shm_atime;         //最后一次映射的时间 __pid_t shm_cpid;             //创建共享内存进程的pid __pid_t shm_lpid;             //最后一次操作共享内存进程的pid shmatt_t shm_nattch;         //共享内存映射的次数
};
struct ipc_perm{__key_t __key;                //ftok获取的key__uid_t uid;                 //用户的ID__gid_t gid;                 //组ID__uid_t cuid;                //创建共享内存的用户的ID__gid_t cgid;                //创建共享内存的组的IDunsigned short int mode;     //消息队列的权限
};

二、信号灯集—控制进程间同步

(一)特点

信号灯集:又叫做信号量数组,他是实现进程间同步的机制

在一个信号灯集中可以有很多个信号灯,这些信号灯之间工作相互互不干扰。
一般使用时使用的都是二值信号灯

(二) 相关API

1. 创建一个信号灯集

#include <sys/sem.h>
int semget(key_t key, int nsems, int semflg);
功能:创建一个信号灯集
参数:key:键值IPC_PRIVATE/keynsems:信号灯集合中信号灯的个数semflag:创建的标志位IPC_CREAT|0666 或 IPC_CREAT|IPC_EXCL|0666
返回值:成功 semid失败 -1  重置错误码
  • 补充:使用IPC_PRIVATE创建的IPC对象, key值属性为0,和IPC对象的编号就没有了对应关系。这样毫无关系的进程,就不能通过key值来得到IPC对象的编号(因为这种方式创建的IPC对象的key值都是0)。因此,这种方式产生的IPC对象,和无名管道类似,不能用于毫无关系的进程间通信。但也不是一点用处都没有,仍然可以用于有亲缘关系的进程间通信。

2. 信号灯集控制函数

int semctl(int semid, int semnum, int cmd, ...);功能:信号灯集的控制函数参数:semid信号灯集的IDsenum:信号灯的编号 从0开始cmd:命令码SETVAL:设置信号灯的值 --->第四个参数val选项GETVAL:获取信号灯的值 --->不需要第四个参数IPC_STAT:获取信号灯集的属性--->第二参数被忽略,第四个参数buf选项IPC_SET :设置信号灯集的属性--->第二参数被忽略,第四个参数buf选项IPC_RMID:删除信号灯集 第二参数被忽略,第4个参数不用填写 @...:可变参union semun{int val; /* Value for SETVAL */struct semid_ds *buf; /* Buffer for IPC_STAT, IPC_SET */};返回值:成功:GETVAL:成功返回信号灯的值其余的命令码成功返回0失败 -1  重置错误码
  • 注:*初始化操作在两个进程中必须只能进行一次,因为进程之间运行是没有顺序的,可以出现其中一个进程已经进行了信号灯集的相关操作后,另一个进程进行初始化而导致出错。

3. 信号灯集的操作函数

int semop(int semid, struct sembuf *sops, size_t nsops);功能:信号灯集中信号灯的操作函数参数:semid:信号灯集的编号sops:操作方式struct sembuf{unsigned short sem_num; //信号灯的编号short sem_op; //操作方式(PV)-1:P操作,申请资源 1:V操作,释放资源short sem_flg; //操作的标志位 0:阻塞 IPC_NOWAIT:非阻塞方式操作}nsops:本次操作信号灯的个数返回值:成功 0失败 -1  重置错误码
  • 注:同时对多个信号灯进行操作时,可以定义一个结构体数组

(三)封装函数

原生函数直接使用会比较繁琐,因此会做二次封装。

初始化(解决可能重复初始化的问题):

int sem_init_pack(key_t key, int nsem){int semid = 0;//不存在创建,存在退出返回semidif(-1 == (semid = semget(key,nsem,IPC_CREAT|IPC_EXCL|0666))){if(EEXIST == errno){//已存在导致的错误if((-1 == (semid = semget(key,nsem,IPC_CREAT|0666)))){return -1;}return semid;}else{//不是已存在导致的错误,说明出错return -1;}}//初始化,第一个置1,其余置0if(semctl(semid,0,SETVAL,1)){return -1;}for(int i = 1; i < nsem; i++){if(semctl(semid,i,SETVAL,0)){return -1;}}return semid;
}

单信号灯PV操作

int sem_p_pack(int semid,int semnum){struct sembuf sembuff={.sem_num=semnum,.sem_op=-1,.sem_flg=0};if(-1 == semop(semid,&sembuff,1)){return errno;}return 0;
}int sem_v_pack(int semid,int semnum){struct sembuf sembuff={.sem_num=semnum,.sem_op=1,.sem_flg=0};if(-1 == semop(semid,&sembuff,1)){return errno;}return 0;
}

多信号灯PV操作
传入一个int型数组指针,数组成员就是要进行p操作的信号灯编号

int sem_p_set_pack(int semid, int *semnum, int num){struct sembuf sembuff[num];for(int i=0; i<num; i++){sembuff[i].sem_flg=1;sembuff[i].sem_op=-1;sembuff[i].sem_num=semnum[i];if(-1 == semop(semid,sembuff+i,1)){return errno;}}return 0;
}int sem_v_set_pack(int semid, int *semnum, int num){if(NULL==semnum) return -1;struct sembuf sembuff[num];for(int i=0; i<num; i++){sembuff[i].sem_flg=1;sembuff[i].sem_op=1;sembuff[i].sem_num=semnum[i];if(-1 == semop(semid,sembuff+i,1)){return errno;}}return 0;
}

销毁

int sem_destroy_pack(int semid){if(-1 == semctl(semid,0,IPC_RMID)){//如果出现这个错误说明是已经销毁过了,无视这个错误if(EINVAL == errno){return 0;}//否则就是出错了,返回错误信息printf("semop error:%s",strerror(errno));return -1;}return 0;
}

这篇关于IO进程线程(十二)进程间通信 共享内存 信号灯集的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

python多进程实现数据共享的示例代码

《python多进程实现数据共享的示例代码》本文介绍了Python中多进程实现数据共享的方法,包括使用multiprocessing模块和manager模块这两种方法,具有一定的参考价值,感兴趣的可以... 目录背景进程、进程创建进程间通信 进程间共享数据共享list实践背景 安卓ui自动化框架,使用的是

Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单

《Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单》:本文主要介绍Springboot的ThreadPoolTaskScheduler线... 目录ThreadPoolTaskScheduler线程池实现15分钟不操作自动取消订单概要1,创建订单后

C语言线程池的常见实现方式详解

《C语言线程池的常见实现方式详解》本文介绍了如何使用C语言实现一个基本的线程池,线程池的实现包括工作线程、任务队列、任务调度、线程池的初始化、任务添加、销毁等步骤,感兴趣的朋友跟随小编一起看看吧... 目录1. 线程池的基本结构2. 线程池的实现步骤3. 线程池的核心数据结构4. 线程池的详细实现4.1 初

Java子线程无法获取Attributes的解决方法(最新推荐)

《Java子线程无法获取Attributes的解决方法(最新推荐)》在Java多线程编程中,子线程无法直接获取主线程设置的Attributes是一个常见问题,本文探讨了这一问题的原因,并提供了两种解决... 目录一、问题原因二、解决方案1. 直接传递数据2. 使用ThreadLocal(适用于线程独立数据)

C#如何优雅地取消进程的执行之Cancellation详解

《C#如何优雅地取消进程的执行之Cancellation详解》本文介绍了.NET框架中的取消协作模型,包括CancellationToken的使用、取消请求的发送和接收、以及如何处理取消事件... 目录概述与取消线程相关的类型代码举例操作取消vs对象取消监听并响应取消请求轮询监听通过回调注册进行监听使用Wa

[Linux]:进程(下)

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:Linux学习 贝蒂的主页:Betty’s blog 1. 进程终止 1.1 进程退出的场景 进程退出只有以下三种情况: 代码运行完毕,结果正确。代码运行完毕,结果不正确。代码异常终止(进程崩溃)。 1.2 进程退出码 在编程中,我们通常认为main函数是代码的入口,但实际上它只是用户级

线程的四种操作

所属专栏:Java学习        1. 线程的开启 start和run的区别: run:描述了线程要执行的任务,也可以称为线程的入口 start:调用系统函数,真正的在系统内核中创建线程(创建PCB,加入到链表中),此处的start会根据不同的系统,分别调用不同的api,创建好之后的线程,再单独去执行run(所以说,start的本质是调用系统api,系统的api

Java IO 操作——个人理解

之前一直Java的IO操作一知半解。今天看到一个便文章觉得很有道理( 原文章),记录一下。 首先,理解Java的IO操作到底操作的什么内容,过程又是怎么样子。          数据来源的操作: 来源有文件,网络数据。使用File类和Sockets等。这里操作的是数据本身,1,0结构。    File file = new File("path");   字

java 进程 返回值

实现 Callable 接口 与 Runnable 相比,Callable 可以有返回值,返回值通过 FutureTask 进行封装。 public class MyCallable implements Callable<Integer> {public Integer call() {return 123;}} public static void main(String[] args

springboot体会BIO(阻塞式IO)

使用springboot体会阻塞式IO 大致的思路为: 创建一个socket服务端,监听socket通道,并打印出socket通道中的内容。 创建两个socket客户端,向socket服务端写入消息。 1.创建服务端 public class RedisServer {public static void main(String[] args) throws IOException {