读者写者问题—内含408真题

读者写者问题—含408

一、问题描述

一个数据问价或记录可以被多个进程共享,我们把只读该文件的进程称为“读者进程”,其他进程为“写者进程”。允许多个进程同时读一个共享对象,但不允许一个写者进程和其他写者进程或读者进程同时访问共享对象。即:保证一个写者进程必须与其他进程互斥的访问共享对象的同步问题;读者-写者问题常用来测试新同步原语。

二、解题思路

首先对于上述问题进行抽象:读者和写者是互斥的,写者和写者是互斥的,读者和读者不互斥;两类进程,一种是写者,另一种是读者。写者很好实现,因为它和其他任何进程都是互斥的,因此对每一个写者进程都给一个互斥信号量的P、V操作即可;而读者进程的问题就较为复杂,它与写者进程是互斥的,而又与其他的读者进程是同步的,因此不能简单的利用P、V操作解决。
下面我们给出三种方案来解决读者和写者之间、读者和读者之间、写者和写者之间的同步与互斥问题:

2.1 读者优先算法(一般意义上的读者写者问题)

为实现Reader和Writer进程之间在读或写时的互斥而设置了一个互斥信号量wmutex。再设置一个整型变量conut表示正在读的进程数目。
仅当count=0时,Reader进程才需要执行P(wmutex)操作;
仅当Reader进程在执行了count减一操作后其值为0时,才需要执行V(wmutex)操作
由于count是一个可被多个Reader进程访问的临界资源,因此为其设置一个互斥信号量cmutex;
其伪码描述如下:

semaphore cmutex=1,wmutex=1;
int count=0;void Reader(){while(1){P(cmutex);if(count==0)P(wmutex);count++;V(cmutex);read;P(cmutex);count--;if(count==0)V(wmutex);V(cmutex);}
}void Writer(){while(1){P(wmutex);write;V(wmutex);}
}

等待Reader进程全部结束之后才逐步执行Writer进程。我们称这样的算法为读者优先算法,也就是说,当存在读进程时,写操作将被延迟,并且只要有一个读进程活跃,随后而来的读进程都将被允许访问文件。这样的方式下,会导致写进程可能长时间等待,且存在写进程“饿死”的情况。


2.2 写者优先算法

所谓写者优先,即:当有读者进程正在执行,写者进程发出申请,这时应该拒绝其他读者进程的请求,等待当前读者进程结束后立即执行写者进程,只有在无写者进程执行的情况下才能够允许读者进程再次运行。
为此,增加一个信号量并且在上面的程序中 writer()和reader()函数中各增加一对PV操作,就可以得到写进程优先的解决程序。

  • 在读者优先的基础上增加信号量rmutex,初值是1,用于禁止所有的读进程。
  • 增加一个记数器,即整型变量writecount,记录写者数,初值是0(原count改为readcount)。
  • writecount为多个写者共享的变量,是临界资源。用互斥信号量wmutex控制, wmutex初值是1
  • mutex用于实现互斥访问缓冲区
    伪码描述如下:
semaphore rmutex=1,cmutex=1,wmutex=1,mutex=1;
int readcount=0,writecount=0;void Reader(){while(1){/**新增代码**/P(rmutex);// 取到该信号量说明此时并没有写进程在等待// 后续读者才可以共享访问临界区// 但是前面已经进入的读进程不受影响/**********/P(cmutex);if(readcount==0)P(mutex);readcount++;V(cmutex);/**新增代码**/V(rmutex);// 已经取到过这个信号量了// 那么就说明已经得到了对临界区的读访问权限// 此时可以一起读/**********/		read;V(cmutex);readcount--;if(readcount==0)V(mutex);V(cmutex);}
}void Writer(){while(1){/**新增代码**/P(wmutex);writecount++;if(writecount==1)// 第一个写进程进来等待以后就P(rmutex);	 // 禁止读进程继续读了V(wmutex);/***********/P(mutex);write;V(mutex);/**新增代码**/P(wmutex);writecount--;if(writecount==0)//当没有写进程时,才允许读进程继续读V(rmutex);V(wmutex);/***********/}
}

2.3 读写公平

为实现读写公平,我们必须要同时满足以下四个条件:

  • 读者写者的优先级相同
  • 读者、写者互斥访问
  • 只允许有一个写者访问临界区
  • 可以有多个读者同时访问临界区的资源
    为此,我们设置一个互斥信号量mutex,其作用是让每个Writer进程和Reader进程进行公平争夺
  • 当Reader争夺到了,那么不管他是不是第一个Reader,他都有权利进入读操作(但是在进行读操作之前,需要释放这个互斥信号量,供后续的Reader和Writer继续公平争夺)
  • 当Writer争夺到了,就等待之前的Reader全部执行完,就可以上处理机运行
semaphore cmutex=1,wmutex=1,mutex=1;
int count=0;void Reader(){while(1){/**新增代码**/P(mutex);//争夺优先权/**********/P(cmutex);if(count==0)P(wmutex);count++;V(cmutex);/**新增代码**/V(mutex);/**********/read;P(cmutex);count--;if(count==0)V(wmutex);V(cmutex);}
}void Writer(){while(1){/**新增代码**/P(mutex);//争夺优先权/**********/P(wmutex);write;V(wmutex);/**新增代码**/V(mutex);/**********/}
}

这个最主要的思路就是让每一次运行的进程都有公平竞争的权利
因为读者优先算法,当读者上处理机运行后,写者就丧失了竞争的权利,只有当读者全部读完才可以重新竞争,这很不公平

2.4 双读者问题(2017年408真题)

先说明一下,双读者问题实际上并不存在,只是针对这道题提出的概念,由于使用常规的读者写者算法会导致并发度不够,所以特地将真题单独作为一个问题考虑

在这里插入图片描述
口说无凭,不如使用对比来更直观地展现其区别吧
首先是使用读者写者问题来解决该问题的算法:

semaphore mutex_x=1;// 对x的互斥访问
semaphore mutex_y=1;// 对y的互斥访问
semaphore mutex_z=1;// 对z的互斥访问
semaphore mutex_count=1;// 对计数器的互斥访问
int count=0;//计数器void thread1(){cnum w;P(mutex_count);if(count==0)P(mutex_y);count++;V(mutex_count);P(mutex_x);w = add(x,y);V(mutex_x);P(mutex_count);count--;if(count==0)V(mutex_y);V(mutex_count);}void thread2(){cnum w;P(mutex_count);if(count==0)P_(mutex_y);count++;V(mutex_count);P(mutex_z);w = add(y,z);V(mutex_z);P(mutex_count);count--;	if(count==0)V(mutex_y);V(mutex_count);	
}void thread3(){cnum w;w.a = 1;w.b = 1;P(mutex_z);z = add(z,w);V(mutex_z);P(mutex_y);y = add(y,w);V(mutex_y);
}

乍一看好像没啥问题,但是有一个很重要的问题是,虽然对于y的访问,实现了读写互斥,同时读与读可以同时进行,但是count信号量限制了并发度,导致两个读操作并不能以最快的方式运行完
下面是标准答案:

semaphore mutex_x=1;// 对x的互斥访问
semaphore mutex_y1=1;// 对y的互斥访问
semaphore mutex_y2=1;// 对y对互斥访问
semaphore mutex_z=1;// 对z的互斥访问
void thread1(){cnum w;P(mutex_y1);P(mutex_x);w = add(x,y);V(mutex_x);V(mutex_y1);
}void thread2(){cnum w;P(mutex_y2);P(mutex_z);w = add(y,z);V(mutex_z);V(mutex_y2);	
}void thread3(){cnum w;w.a = 1;w.b = 1;P(mutex_z);z = add(z,w);V(mutex_z);P(mutex_y1);// 只有同时拥有互斥信号量P(mutex_y2);// 才算是获取了访问权限y = add(y,w);V(mutex_y1);V(mutex_y2);
}

上面的代码,可以大大提高并发度,因为1,2两个线程也可以保证在读y时是绝对并行的

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

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

相关文章

点亮一个LED+LED闪烁+LED流水灯——“51单片机”

各位CSDN的uu们好呀,这是小雅兰的最新专栏噢,最近小雅兰学习了51单片机的知识,所以就想迫不及待地分享出来呢!!!下面,让我们进入51单片机的世界吧!!! 点亮一个…

Linux基础命令汇总

用户管理 su 切换用户:su 用户名 logname 显示当前用户的登录用户名:logname useradd 创建用户:useradd 用户名创建用户时指定用户的主组:useradd -g 组名 用户名 usermod 添加附属组:usermod -G 组…

2023年8月嵌入式项目开发专题总汇

一、前言 本文介绍基于嵌入式系统和C语言开发的系列项目。这些项目涵盖了多个领域,从自动化控制到游戏开发,从计算机网络到物联网应用。通过这些项目的开发过程,将深入探讨各种技术和解决方案,并分享相关经验和知识。 在本文中&…

cesium 雷达扫描 (线行扩散效果)

cesium 雷达扫描 (线行扩散效果) 1、实现方法 使用ellipse方法加载圆型,修改ellipse中material方法来实现效果 2、示例代码 2.1、 <!DOCTYPE html> <html lang="en"><head><<

分类预测 | Matlab实现BES-ELM秃鹰搜索算法优化极限学习机分类预测

分类预测 | Matlab实现BES-ELM秃鹰搜索算法优化极限学习机分类预测 目录 分类预测 | Matlab实现BES-ELM秃鹰搜索算法优化极限学习机分类预测分类效果基本描述程序设计参考资料 分类效果 基本描述 Matlab实现BES-ELM秃鹰搜索算法优化极限学习机分类预测&#xff08;完整源码和数…

【CVPR 2023】DSVT: Dynamic Sparse Voxel Transformer with Rotated Sets

文章目录 开场白效果意图 重点VoxelNet: End-to-End Learning for Point Cloud Based 3D Object DetectionX-Axis DSVT LayerY-Axis DSVT Layer Dynamic Sparse Window AttentionDynamic set partitionRotated set attention for intra-window feature propagation.Hybrid wind…

优维低代码实践:应用级配置

优维低代码技术专栏&#xff0c;是一个全新的、技术为主的专栏&#xff0c;由优维技术委员会成员执笔&#xff0c;基于优维7年低代码技术研发及运维成果&#xff0c;主要介绍低代码相关的技术原理及架构逻辑&#xff0c;目的是给广大运维人提供一个技术交流与学习的平台。 优维…

[NOIP2012 提高组] 开车旅行

[NOIP2012 提高组] 开车旅行 题目描述 小 A \text{A} A 和小 B \text{B} B 决定利用假期外出旅行&#xff0c;他们将想去的城市从 $1 $ 到 n n n 编号&#xff0c;且编号较小的城市在编号较大的城市的西边&#xff0c;已知各个城市的海拔高度互不相同&#xff0c;记城市 …

亚信科技AntDB数据库 高并发、低延迟、无死锁,深入了解AntDB-M元数据锁的实现

AntDB-M在架构上分为两层&#xff0c;服务层和存储引擎层。元数据的并发管理集中在服务层&#xff0c;数据的存储访问在存储引擎层。为了保证DDL操作与DML操作之间的一致性&#xff0c;引入了元数据锁&#xff08;MDL&#xff09;。 AntDB-M提供了丰富的元数据锁功能&#xff0…

Koa处理请求数据

在开发中&#xff0c;后端接收到请求参数后&#xff0c;需要解析参数。请求分为很多种类型&#xff0c;比如常见的get和post。 请求参数 Koa本身可以解析get请求参数&#xff0c;不能解析post请求参数。例如&#xff1a; router.get(/api/get/userInfo, async (context) >…

新手--安装好Quartus II13.0(带modelsim集成包)并用Quartus II搭建一个工程

前言 今天是国庆节&#xff0c;我们正式来学习Quartus II13.0软件的安装与使用。学习verilog与学习C语言都是学习一门语言&#xff0c;那么学习一门语言&#xff0c;光看理论不敲代码绝对是学习不好的。要用verilog语言敲代码&#xff0c;就要像C语言那样搭建起语言的编译环境&…

C语言 Cortex-A7核 IIC实验

iic.h #ifndef __IIC_H__ #define __IIC_H__ #include "stm32mp1xx_gpio.h" #include "stm32mp1xx_rcc.h" /* 通过程序模拟实现I2C总线的时序和协议* GPIOF ---> AHB4* I2C1_SCL ---> PF14* I2C1_SDA ---> PF15** */#define SET_SDA_OUT do{…

B. Comparison String

题目&#xff1a; 样例&#xff1a; 输入 4 4 <<>> 4 >><< 5 >>>>> 7 <><><><输出 3 3 6 2 思路&#xff1a; 由题意&#xff0c;条件是 又因为要使用尽可能少的数字&#xff0c;这是一道贪心题&#xff0c;所以…

Linux CentOS7 vim临时文件

在vim中&#xff0c;由于断网、停电、故意退出、不小心关闭终端等多种原因&#xff0c;正在编辑的文件没有保存&#xff0c;系统将会为文件保存一个交换文件&#xff0c;或称临时文件&#xff0c;或备份文件。 如果因某种原因产生了交换文件&#xff0c;每次打开文件时&#x…

详解分布式搜索技术之elasticsearch

目录 一、初识elasticsearch 1.1什么是elasticsearch 1.2elasticsearch的发展 1.3为什么学习elasticsearch? 1.4正向索引和倒排索引 1.4.1传统数据库采用正向索引 1.4.2elasticsearch采用倒排索引 1.4.3posting list ​1.4.4总结 1.5 es的一些概念 1.5.1文档和字段 …

鞋类 整鞋试验方法 剥离强度

声明 本文是学习GB-T 3903.3-2011 鞋类 整鞋试验方法 剥离强度. 而整理的学习笔记,分享出来希望更多人受益,如果存在侵权请及时联系我们 1 范围 GB/T 3903 的本部分规定了整鞋鞋底与鞋帮或外底与外中底之间剥离强度的试验方法。 本部分适用于采用模压、硫化、注塑、灌注、胶…

C进阶--字符函数和字符串函数介绍

✨ 更多细节参考 cplusplus.com/reference/cstring/ 使用方式&#xff1a; ⭕ 求字符串长度 &#x1f58c; strlen 函数原型&#xff1a; size_t strlen ( const char * str ); 作用&#xff1a; 获取字符串长度 ✨补充&#xff1a; ⭐字符串以 \0 作为结束标志&…

5.外部中断

中断初始化配置步骤&#xff1a; IO口初始化配置 开启中断总允许EA 打开某个IO口的中断允许 打开IO口的某一位的中断允许 配置该位的中断触发方式 中断函数&#xff1a; #pragma vector PxINT_VECTOR __interrupt void 函数名(void){}#pragma vector PxINT_VECTOR __int…

喝健康白酒 有益生心健康

中国的制酒史源远流长&#xff0c;酒渗透在中华五千年的文化中。酒与烟不同&#xff0c;烟对人体有百害而无一利&#xff0c;而对于酒&#xff0c;若掌握好饮酒的度&#xff0c;对人体有一定的养生作用&#xff0c;所以我们通常会说“戒烟限酒”。 据一些专家研究&#xff0c;…

云原生Kubernetes:对外服务之 Ingress

目录 一、理论 1.Ingress 2.部署 nginx-ingress-controller(第一种方式) 3.部署 nginx-ingress-controller(第二种方式) 二、实验 1.部署 nginx-ingress-controller(第一种方式) 2.部署 nginx-ingress-controller(第二种方式) 三、问题 1.启动 nginx-ingress-controll…