PV操作解决经典进程同步问题

news2025/1/11 7:07:15

一.经典同步问题

在学习《操作系统》时,会接触到进程的概念,其中不可避免的接触到进程同步问题,今天我们用熟悉的PV操作解决一些经典的进程同步问题。

二.生产者-消费者问题

1.问题描述

问题描述:一组生产者进程和一组消费者进程共享一个初始为空、大小为n的缓冲区,只有缓冲区没满时,生产者才能把消息放入缓冲区,否则必须等待;只有缓冲区不空时,消费者才能从中取出消息,否则必须等待。由于缓冲区是临界资源,它只允许一个生产者放入消息,或一个消费者从中取出消息。

2.问题分析

  • 1)关系分析。生产者和消费者对缓冲区互斥访问是互斥关系,同时生产者和消费者又是一个相互协作的关系,只有生产者生产之后,消费者才能消费,它们也是同步关系。
  • 2)整理思路。这里比较简单,只有生产者和消费者两个进程,正好是这两个进程存在着互斥关系和同步关系。那么需要解决的是互斥和同步PV操作的位置。
  • 3)信号量设置。信号量mutex 作为互斥信号量,用于控制互斥访问缓冲池,互斥信号量初值为1;信号量full用于记录当前缓冲池中的“满”缓冲区数,初值为0。信号量empty用于记录当前缓冲池中的“空”缓冲区数,初值为n。

3.PV描述

生产者-消费者进程的描述如下:

semaphore mutex=1 ;            //临界区互斥信号量
semaphore empty=n;             //空闲缓冲区
semaphore full=0;              //缓冲区初始化为空

producer() {                  //生产者进程
	while (1) {
		produce an item in nextp;      //生产数据
		P (empty); (要用什么, p一下)    //获取空缓冲区单元
		P (mutex); (互斥夹紧)           //进入临界区
		add nextp to buffer; (行为)    //将数据放入缓冲区
		V (mutex) ;(互斥夹紧)           //离开临界区,释放互斥信号
		V(full); (提供什么, V一下)      //满缓冲区数加1
    }
}

consumer() {                         //消费者进程
	while(1) {
		P (full) ;                   //获取满缓冲区单元
		P (mutex) ;                         //进入临界区
		remove an item from buffer;        //从缓冲区中取出数据
		V (mutex) ;                        //离开临界区,释放互斥信号量
		V (empty) ;                        //空缓冲区数加1
		consume the item;                  //消费数据
    }
}

三.复杂生产者-消费者问题

1.问题描述

问题描述:桌子上有一个盘子,每次只能向其中放入-一个水果。爸爸专向盘子中放苹果,妈妈专向盘子中放橘子,儿子专等吃盘子中的橘子,女儿专等吃盘子中的苹果。只有盘子为空时,爸爸或妈妈才可向盘子中放-一个水果;仅当盘子中有自己需要的水果时,儿子或女儿可以从盘子中取出。

2.问题分析

1)关系分析。这里的关系要稍复杂一些。由每次只能向其中放入一只水果可知,爸爸和妈妈是互斥关系。爸爸和
女儿、妈妈和儿子是同步关系,而且这两对进程必须连起来,儿子和女儿之间没有互斥和同步关系,因为他们是选择条件执行,不可能并发,如图所示。

在这里插入图片描述

2)整理思路。这里有4个进程,实际上可抽象为两个生产者和两个消费者被连接到大小为1的缓冲区上。

3)信号量设置。首先将信号量plate 设置互斥信号量,表示是否允许向盘子放入水果,初值
为1表示允许放入,且只允许放入一个。信号量apple表示盘子中是否有苹果,初值为0
表示盘子为空,不许取,apple= 1表示可以取。信号量orange表示盘子中是否有橘子,
初值为0表示盘子为空,不许取,orange= 1表示可以取。

3.PV描述

解决该问题的代码如下:

semaphore plate=1, apple=0, orange=0;

dad() {           //父亲进程
	while(1) {
		prepare an apple;
		P (plate) ;							//互斥向盘中取、放水果
		put the apple on the plate;			//向盘中放苹果
		V(apple) ;							//允许取苹果
    }
}

mom() {			//母亲进程
	while (1) {
		prepare an orange;
		P(plate) ;							//互斥向盘中取、放水果
		put the orange on the plate;		//向盘中放橘子
		V (orange) ;						//允许取橘子
    }
}

son() {			//儿子进程
	while (1) {
		P (orange) ;						//互斥向盘中取橘子
		take an orange from the plate;
		V(plate) ;							//允许向盘中取、放水果
		eat the orange;
    }
}

daughter() {	//女儿进程
	while(1) {
		P (apple) ;							// 互斥向盘中取苹果
		take an apple from the plate;
		V(plate) ;							//允许向盘中取、放水果
        eat the apple ;
    }
}


四.读者-写者问题

1.问题描述

问题描述:有读者和写者两组并发进程,共享-一个文件,当两个或以上的读进程同时访问共享数据时不会产生副作用,但若某个写进程和其他进程(读进程或写进程)同时访问共享数据时,则可能导致数据不一致的错误。因此要求:①允许多个读者可以同时对文件执行读操作;②只允许一个写者往文件中写信息;③任意一 个写者在完成写操作之前不允许其他读者或写者工作;④写者执行写操作前,应让已有的读者和写者全部退出。

2.问题分析

1)关系分析。由题目分析读者和写者是互斥的,写者和写者也是互斥的,而读者和读者不存在互斥问题。

2)整理思路。两个进程,即读者和写者。写者是比较简单的,它和任何进程互斥,用互斥信号量的P操作、V操作即可解决。读者的问题比较复杂,它必须在实现与写者互斥的同时,实现与其他读者的同步,因此简单的一对P操作、V操作是无法解决问题的。这里用到了一个计数器,用它来判断当前是否有读者读文件。当有读者时,写者是无法写文件的,此时读者会一直占用文件, 当没有读者时,写者才可以写文件。同时,这里不同读者对计数器的访问也应该是互斥的。

3)信号量设置。首先设置信号量count为计数器,用于记录当前读者的数量,初值为0;设置mutex为互斥信号量,用于保护更新count变量时的互斥;设置互斥信号量rw,用于保证读者和写者的互斥访问。

3.PV描述

代码如下:

int count=0;						//用于记录当前的读者数量
semaphore mutex=1 ;					//用于保护更新count变量时的互斥
semaphore rw=1;						//用于保证读者和写者互斥地访问文件

writer(){
	while(1) {
		P (rw) ;					//互斥访问共享文件
		writing;					//写入
		V (rw) ;					//释放共享文件
    }
}

reader () {							//读者进程
	while(1) {
		P (mutex) ;					//互斥访问count变量
		if (count==0)				//当第一个读进程读共享文件时
			P (rw) ;				//阻止写进程写
		count++ ;					//读者计数器加1
		V (mutex) ;					//释放互斥变量count
		reading;					//读取
		P (mutex) ;					//互斥访问count变量
		count-- ;					//读者计数器减1
		if (count==0)				//当最后一一个读进程读完共享文件
			V(rw) ;					//允许写进程写
        V (mutex) ;					//释放互斥变量count
    }
}



五.哲学家进餐问题

1.问题描述

问题描述:一张圆桌边 上坐着5名哲学家,每两名哲学家之间的桌上摆一根筷子, 两根筷子中间是一碗米饭。哲学家们倾注毕生精力用于思考和进餐,哲学家在思考时,并不影响他人。只有当哲学家饥饿时,才试图拿起左、右两根筷子(一根一根地拿起)。若筷子已在他人手,则需要等待。饥饿的哲学家只有同时拿到了两根筷子才可以开始进餐,进餐完毕后,放下筷子继续思考。

2.问题分析

1)关系分析。5名哲学家与左右邻居对其中间筷子的访问是互斥关系。

2)整理思路。显然,这里有5个进程。本题的关键是如何让一名哲学家拿到左右两根筷子而不造成死锁或饥饿现象。解决方法有两个:一是让他们同时拿两根筷子;二是对每名哲学家的动作制定规则,避免饥饿或死锁现象的发生。

3)信号量设置。定义互斥信号量数组chopstick[5]={1,1,1,1,1},用于对5个筷子的互斥访问。哲学家按顺序编号为0~4,哲学家i左边筷子的编号为i,哲学家右边筷子的编号为(i + 1)%5。

3.PV描述

为防止死锁发生,可对哲学家进程施加一些限制条件, 比如至多允许4名哲学家同时进餐;仅当一名哲学家左右两边的筷子都可用时,才允许他抓起筷子;对哲学家顺序编号,要求奇数号哲学家先拿左边的筷子,然后拿右边的筷子,而偶数号哲学家刚好相反。制定的正确规则如下:假设采用第二种方法,当一名哲学家左右两边的筷子都可用时,才允许他抓起筷子。

semaphore chopstick[5]={1,1,1,1,1};			 //初始化信号量
semaphore mutex=1 ;							//设置取筷子的信号量

Pi(){										//i号哲学家的进程
	do {
		P (mutex) ;							//在取筷子前获得互斥量
		P (chopstick[i]) ;					//取左边筷子
		P (chopstick[ (i+1)85]) ;			//取右边筷子
		V (mutex) ;							//释放取筷子的信号量
		eat;								//进餐
        V (chopstick[i]) ;					//放回左边筷子
		V (chopstick[ (i+1)85]) ;			//放回右边筷子
		think;								//思考
	} while(l) ;
}


六.吸烟者问题

1.问题描述

问题描述:假设一个系统有三个抽烟者进程和一个供应者进程。每个抽烟者不停地卷烟并抽掉它,但要卷起并抽掉一支烟, 抽烟者需要有三种材料:烟草、纸和胶水。三个抽烟者中,第一个拥有烟草,第二个拥有纸,第三个拥有胶水。供应者进程无限地提供三种材料,供应者每次将两种材料放到桌子上,拥有剩下那种材料的抽烟者卷一根烟并抽掉它, 并给供应者一个信 号告诉已完成,此时供应者就会将另外两种材料放到桌上,如此重复(让三个抽烟者轮流地抽烟)。

2.问题分析

问题分析:

  • 1)关系分析。供应者与三个抽烟者分别是同步关系。由于供应者无法同时满足两个或以上的抽烟者, 三个抽烟者对抽烟这个动作互斥( 或由三个抽烟者轮流抽烟得知)。
  • 2)整理思路。显然这里有4个进程。供应者作为生产者向三个抽烟者提供材料。
  • 3)信号量设置。信号量offer1, offer2, offer3分别表示烟草和纸组合的资源、烟草和胶水组合的资源、纸和胶水组合的资源。信号量finish用于互斥进行抽烟动作。

3.PV描述

代码如下:

int num=0;				//存储随机数
semaphore offer1=0;		//定义信号量对应烟草和纸组合的资源
semaphore offer2=0;		//定义信号量对应烟草和胶水组合的资源
semaphore offer3=0;		//定义信号量对应纸和胶水组合的资源
semaphore finish=0;		//定义信号量表示抽烟是否完成

process P1() {		//供应者
	while(1) {
		num++ ;
		num=num%3;
		if (num==0)
			V(offer1) ;				//提供烟草和纸
		else if (num==1)
			V (offer2) ;			//提供烟草和胶水
		else
			V(offer3) ;				//提供纸和胶水
		任意两种材料放在桌子上;
		P (finish) ;
    }
}

process P2() {		//拥有烟草者
	while(1) {
		P(offer3) ;
		拿纸和胶水,卷成烟,抽掉;
		V(finish) ;
    }
}

process P3() {		//拥有纸者
	while (1) {
		P(offer2) ;
		拿烟草和胶水,卷成烟,抽掉;
		V(finish) ;
    }
}

process P4() {		//拥有胶水者
	while (1) {
		P (offerl) ;
		拿烟草和纸,卷成烟,抽掉;
		V(finish) ;
    }
}


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

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

相关文章

Python-正则表达式(给同事培训篇2)

本篇依旧是我在公司给同事培训的内容,与上一篇类似,不过本篇会内容偏简单,会多讲两种。一起看看; 数据 datas """ [16:45:18]2 [16:45:18]# cap: 13, 12, 11 [16:45:18]cap: 13, 12, 11 [16:45:18]cap: 13, 12, 1…

elementUI this.$confirm 文字大小样式

dangerouslyUseHTMLString:true // message部分 以html片段处理 customClass //MessageBox 的自定义类名 整个comfirm框自定义类名 cancelButtonClass // 取消按钮的自定义类名 confirmButtonClass // 确定按钮的自定义类名<style> .addcomfirm{width: 500px; } .a…

IO进、线程——标准文件IO和时间函数

1.文件IO 最直观的系统调用 1.1打开文件 int open(const char *pathname, int flags, mode_t mode);功能&#xff1a;打开/创建后打开一个文件 返回值&#xff1a;成功返回文件描述符&#xff0c;失败-10 —— 标准输入 1 —— 标准输出 2 —— 标准出错参数说明&#xf…

Hive之窗口函数lag()/lead()

一、函数介绍 lag()与lead函数是跟偏移量相关的两个分析函数 通过这两个函数可以在一次查询中取出同一字段的前N行的数据(lag)和后N行的数据(lead)作为独立的列,从而更方便地进行进行数据过滤&#xff0c;该操作可代替表的自联接&#xff0c;且效率更高 lag()/lead() lag(c…

VS构建项目报错信息及解决办法05

报错信息及解决8&#xff1a; 报错信息详情&#xff1a;无法解析的外部符号“__iob_func” 原因&#xff1a;因VS不同版本之间对stdin,stdout,stder的定义不同&#xff0c;导致不同VS版本之间无法正确的调用函数。 eg: * 当libjpeg-turbo为vs2010编译时&#xff0c;vs2015下…

qsort的使用及模拟实现

qsort函数是C语言库中提供的一种快速排序&#xff0c;头文件是stdlib.h qsort的使用 qsort函数需要四个参数&#xff1a; 1.排序的起始位置的地址&#xff08;数组名&#xff09;: arr 2.排序元素的个数&#xff1a; sizeof&#xff08;arr)/sizeof(arr[0]) 3.排序元素…

使用BERT分类的可解释性探索

最近尝试了使用BERT将告警信息当成一个文本去做分类&#xff0c;从分类的准召率上来看&#xff0c;还是取得了不错的效果&#xff08;非结构化数据强标签训练&#xff0c;BERT确实是一把大杀器&#xff09;。但准召率并不是唯一追求的目标&#xff0c;在安全场景下&#xff0c;…

java中线程池、Lambda表达式、file类、递归

线程池&#xff1a; 在多线程的使用过程中&#xff0c;会存在一个问题&#xff1a;如果并发的线程数量很多&#xff0c;并且每个线程都执行一个时间很短的任务就结束&#xff0c;这样频繁的创建线程就会大大降低系统的效率&#xff0c;因为线程的创建和销毁都需要时间。 线程…

maven编译报错

参考链接&#xff1a;mvn打包No compiler is provided in this environment. Perhaps you are running on a JRE rather than a JDK_51CTO博客_mvn打包命令 在执行 yum install -y java-1.8.0-opensdk命令后&#xff0c;使用maven去编译打包&#xff0c;结果报错&#xff0c; …

体渲染光线行进算法【NeRF必读】

为了积分由于内散射而沿射线产生的入射光&#xff0c;我们将射线穿过的体块分解为小体块元素&#xff0c;并将每个小体块元素对整个体块对象的贡献结合起来&#xff0c;有点像我们在 2D 编辑软件&#xff08;例如 Photoshop&#xff09;中将带有遮罩或 Alpha 通道&#xff08;通…

ClickHouse(三):ClickHouse单节点搭建

进入正文前&#xff0c;感谢宝子们订阅专题、点赞、评论、收藏&#xff01;关注IT贫道&#xff0c;获取高质量博客内容&#xff01; &#x1f3e1;个人主页&#xff1a;含各种IT体系技术,IT贫道_Apache Doris,Kerberos安全认证,随笔-CSDN博客 &#x1f4cc;订阅&#xff1a;拥抱…

pytorch学习——线性神经网络——1线性回归

概要&#xff1a;线性神经网络是一种最简单的神经网络模型&#xff0c;它由若干个线性变换和非线性变换组成。线性变换通常表示为矩阵乘法&#xff0c;非线性变换通常是一个逐元素的非线性函数。线性神经网络通常用于解决回归和分类问题。 一.线性回归 线性回归是一种常见的机…

WebGPU(八):三角形渲染

WebGPU(八)&#xff1a;三角形渲染 三角形的渲染其实很简单&#xff0c;只是需要设置很详细的render pipeline以及shader。 // Select which render pipeline to use wgpuRenderPassEncoderSetPipeline(renderPass, pipeline); // Draw 1 instance of a 3-vertices shape wgp…

C# 全局响应Ctrl+Alt+鼠标右键

一、简述 某些应用&#xff0c;我们希望全局自定义热键。按键少了会和别的应用程序冲突&#xff0c;按键多了可定用户操作不变。因此我计划左手用CtrlAlt&#xff0c;右手用鼠标右键呼出我自定义的菜单。 我使用键盘和鼠标事件进行简单测试&#xff08;Ctrl鼠标右键&#xff…

TypeScript -- 函数

文章目录 TypeScript -- 函数JS -- 函数的两种表现形式函数声明函数的表达式es6 箭头函数 TS -- 定义一个函数TS -- 函数声明使用接口(定义)ts 定义参数可选参数写法 -- ?的使用TS函数 -- 设置剩余参数函数重载 TypeScript – 函数 JS – 函数的两种表现形式 我们熟知js有两…

MySQLExplain详解

Explain使用场景 查询性能优化&#xff1a;EXPLAIN可以帮助开发者分析查询语句的执行计划&#xff0c;判断是否有效地使用了索引、是否有可能导致全表扫描等性能问题。通过EXPLAIN的输出&#xff0c;可以找到潜在的性能瓶颈&#xff0c;并优化查询语句、创建合适的索引或调整表…

Win11虚拟机安装并使用

windows11 虚拟机安装 操作如下&#xff1a;1.进入微软官网2.打开虚拟机应用创建新虚拟机3.选择刚下载IOS文件4 设置虚拟机磁盘空间大小&#xff0c;这个数字可以随便写&#xff0c;反正都是虚拟的&#xff0c;但不可以低于64GB。下面的是否拆分磁盘文件&#xff0c;可更具需要…

大数据课程C4——ZooKeeper结构运行机制

文章作者邮箱&#xff1a;yugongshiyesina.cn 地址&#xff1a;广东惠州 ▲ 本章节目的 ⚪ 了解Zookeeper的特点和节点信息&#xff1b; ⚪ 掌握Zookeeper的完全分布式安装 ⚪ 掌握Zookeeper的选举机制、ZAB协议、AVRO&#xff1b; 一、Zookeeper-简介 1. 特点…

【计网】什么是三次握手四次挥手

文章目录 1、什么是TCP2、什么是TCP连接2.1、连接概念2.2、如何唯一确定一个TCP连接2.3、TCP最大连接数 3、三次握手3.1、为什么需要三次握手3.2、三次握手过程3.3、为什么一定是三次3.3.1、避免历史连接3.3.2、同步双方初始序列号3.3.3、避免资源浪费3.3.4、总结 3.4、握手丢失…

vue实现卡牌数字动态翻牌效果

vue实现卡牌数字动态翻牌效果 1. 实现效果2. 实现代码 1. 实现效果 在大屏项目中&#xff0c;我们尝尝会遇到卡牌式数字显示且能动态翻牌的效果&#xff0c;效果图如下&#xff1a; 2. 实现代码 <template><div class"days-box"><div class"op…