38、IO进程线程/共享内存、信号灯集实现进程间同步通信20240226

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

一、创建共享内存、和信号灯集实现两个进程间同步通信(进程1先发送,进程2后接收,待2接收完成后,1才能继续发送(用信号灯集实现同步机制,有顺序的先后执行))。

代码:

1信号灯集封装函数头文件:sem.h

#ifndef _SEM_H__
#define _SEM_H__//创建或打开信号灯集:参数为要申请的信号灯集包含的灯的个数,返回信号灯集id
int open_sem(int semcount);//进行申请资源操作:参数为要申请的信号灯集id,以及要申请灯的编号
int P(int semid,int semno);//释放资源操作:参数为要释放的信号灯集id,以及要释放灯的编号
int V(int semid,int semno);//删除信号灯集:参数为要删除的信号灯集id
int del_sem(int semid);#endif

2信号灯集创建、初始化、申请、释放、删除操作

#include<myhead.h>
union semun {int              val;    /* Value for SETVAL */struct semid_ds *buf;    /* Buffer for IPC_STAT, IPC_SET */unsigned short  *array;  /* Array for GETALL, SETALL */struct seminfo  *__buf;  /* Buffer for IPC_INFO(Linux-specific) */
};int init_semno(int semid,int semno)
{union semun buf;printf("请输入要给编号为%d的灯设置的值:",semno);scanf("%d",&buf.val);getchar();//调用控制函数完成初始化if(semctl(semid,semno,SETVAL,buf)==-1){perror("semctl error");return -1;}return 0;
}
//创建或打开信号灯集:参数为要申请的信号灯集包含的灯的个数,返回信号灯集id
int open_sem(int semcount)
{key_t key=-1;if((key=ftok("/",'s'))==-1){perror("ftok error");return -1;}int semid=-1;//根据key值创建信号灯集if((semid=semget(key,semcount,IPC_CREAT|IPC_EXCL|0664))==-1){if(errno==EEXIST)//如果文件已经存在则直接打开返回semid{semid=semget(key,semcount,IPC_CREAT|0664);return semid;}perror("semget error");return -1;}for(int i=0;i<semcount;i++)//首次打开需要初始化信号灯集{init_semno(semid,i);//对编号为i的灯进行初始化}return semid;
}//进行申请资源操作:参数为要申请的信号灯集id,以及要申请灯的编号
int P(int semid,int semno)
{struct sembuf buf;buf.sem_num=semno;buf.sem_op=-1;buf.sem_flg=1;if(semop(semid,&buf,1)==-1){perror("semop error");return -1;}return 0;
}//释放资源操作:参数为要释放的信号灯集id,以及要释放灯的编号
int V(int semid,int semno)
{	struct sembuf buf;buf.sem_num=semno;buf.sem_op=1;buf.sem_flg=1;if(semop(semid,&buf,1)==-1){perror("semop error");return -1;}return 0;
}//删除信号灯集:参数为要删除的信号灯集id
int del_sem(int semid)
{if(semctl(semid,0,IPC_RMID)==-1){perror("delete error");return -1;}
}

3负责发送信息的进程(使用共享内存)

#include<myhead.h>
#include"sem.h"
#define PAGE_SIZE 4096
int main(int argc, const char *argv[])
{int semid=open_sem(2);//创建信号灯集key_t key=-1;//创建key值if((key=ftok("/",'m'))==-1){perror("ftok error");return -1;}int shmid=-1;//根据key创建共享内存返回共享内存idif((shmid=shmget(key,PAGE_SIZE,IPC_CREAT|0664))==-1){perror("shmget error");return -1;}char *addr=shmat(shmid,NULL,0);//根据shmid将共享内存映射到用户空间if(addr==(void*)-1){perror("shmat error");return -1;}//使用共享内存写入操作while(1){//申请0号灯资源P(semid,0);printf("请输入>>>");fgets(addr,PAGE_SIZE,stdin);addr[strlen(addr)-1]=0;printf("发送成功 \n");//释放1号灯资源V(semid,1);if(strcmp(addr,"quit")==0)break;}//取消映射if(shmdt(addr)==-1){perror("shmdt error");return -1;}//删除共享内存if(shmctl(shmid,IPC_RMID,NULL)==-1){perror("shmctl error");return -1;}return 0;
}

 4负责接收信息的进程

#include<myhead.h>
#include"sem.h"
#define PAGE_SIZE 4096
int main(int argc, const char *argv[])
{//创建信号灯集int semid=open_sem(2);key_t key=-1;//创建key值if((key=ftok("/",'m'))==-1){perror("ftok error");return -1;}int shmid=-1;//根据key创建共享内存返回共享内存idif((shmid=shmget(key,PAGE_SIZE,IPC_CREAT|0664))==-1){perror("shmget error");return -1;}char *addr=shmat(shmid,NULL,0);//根据shmid将共享内存映射到用户空间if(addr==(void*)-1){perror("shmat error");return -1;}//使用共享内存读取操作while(1){//申请1号灯资源P(semid,1);printf("收到数据%s\n",addr);if(strcmp(addr,"quit")==0)break;V(semid,0);//释放0号灯资源}//取消映射if(shmdt(addr)==-1){perror("shmdt error");return -1;}//删除信号灯集del_sem(semid);return 0;
}

运行:

 思维导图:

这篇关于38、IO进程线程/共享内存、信号灯集实现进程间同步通信20240226的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

基于MySQL Binlog的Elasticsearch数据同步实践

一、为什么要做 随着马蜂窝的逐渐发展,我们的业务数据越来越多,单纯使用 MySQL 已经不能满足我们的数据查询需求,例如对于商品、订单等数据的多维度检索。 使用 Elasticsearch 存储业务数据可以很好的解决我们业务中的搜索需求。而数据进行异构存储后,随之而来的就是数据同步的问题。 二、现有方法及问题 对于数据同步,我们目前的解决方案是建立数据中间表。把需要检索的业务数据,统一放到一张M

服务器集群同步时间手记

1.时间服务器配置(必须root用户) (1)检查ntp是否安装 [root@node1 桌面]# rpm -qa|grep ntpntp-4.2.6p5-10.el6.centos.x86_64fontpackages-filesystem-1.41-1.1.el6.noarchntpdate-4.2.6p5-10.el6.centos.x86_64 (2)修改ntp配置文件 [r

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识