哲学家就餐问题及其实现

2023-12-05 10:38
文章标签 实现 问题 哲学家 就餐

本文主要是介绍哲学家就餐问题及其实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

哲学家就餐问题描述

哲学家就餐问题是指,有五个哲学家围坐一桌,每两个哲学家之间都有一只叉子,一共有五只叉子。每个哲学家都只有两个动作,即思考和就餐,哲学家思考的时候不需要任何的资源,但只有同时拿起他左右的两只叉子,才能开始进餐。进餐完毕后将叉子放归原位。这个问题在于,应该如何保证哲学家们的动作有序进行,如不会出现有人永远拿不到叉子的情况。

一些初步的尝试

第一次尝试

对哲学家就餐问题进行分析,可以发现,相邻的两个哲学家对他们中间的叉子应该是互斥访问的。为此,为每个叉子设置一个信号量,每个哲学家在进餐之间,首先需要获得其左右两个叉子,即分别对其左右的两只叉子做一次P操作;进餐完毕后,将两只叉子放归原位,即分别对其左右两只叉子做一次V操作。实现的伪代码如下:

Semaphore forks[5];
for(fork : forks)fork.sem = 1;				//initialize to 1#define LEFT  (i - 1 + 5) % 5
#define RIGHT (i + 1) % 5//for philosopher i
void philosopher(){while(true){thinking();forks[LEFT].P();forks[RIGHT].V();eating();forks[LEFT].V();forks[RIGHT].V();}
}

容易看出,这种实现方法是有问题的,倘若所有五个哲学家同时想要进餐,有一种情况是他们都分别拿起了他们左边的叉子,这样所有哲学家想要获得右边的叉子时都会失败而进入阻塞状态,并且这种阻塞将一直进行下去,因为没有哲学家会主动释放已经获得的叉子,即出现了死锁。

第二次尝试

为了对上面的尝试做出改进,可以注意到,在任意时刻只可能有两个哲学家在同时进餐,因此可以做一些额外的限制,使得只能有两个哲学家同时请求获得叉子,为此可以再设置一个初值为2的信号量mutex,实现的伪代码如下:

Semaphore forks[5];
for(fork : forks)fork.sem = 1;
Semaphore mutex(2);//for philosopher i
void philosopher(){thinking();//I'm hungrymutex.P();forks[LEFT].P();forks[RIGHT].P();eating();forks[LEFT].V();forks[RIGHT].V();mutex.V();
}

由于任意时刻只能有两个哲学家在请求叉子,因此至少会有一个哲学家可以同时获得他左右的两只叉子然后开始进餐,第一次尝试中死锁的情况在这里不会再发生。但是如果两个请求叉子的哲学家是相邻的,他们中必有一个会进入阻塞状态,此时另外两只叉子还是空闲的,却不能有哲学家进入临界区获得叉子了,也就是说这种策略会导致资源的浪费。

第三次尝试

再次对第一次尝试进行分析,可以发现,第一次尝试之所以会失败,是因为所有哲学家都请求了同一侧的叉子,导致出现了循环等待的情况。为了解决这个问题,可以让不同的哲学家请求不同侧的叉子,比如奇数号的哲学家优先请求左侧叉子,而偶数号的哲学家优先请求右侧叉子。这种策略的伪代码如下:

Semaphore forks[5];
for(fork : forks)fork.sem = 1;				//initialize to 1#define LEFT  (i - 1 + 5) % 5
#define RIGHT (i + 1) % 5//for philosopher i
void philosopher(){while(true){thinking();if(i % 2 == 1){forks[LEFT].P();forks[RIGHT].V();}else{forks[RIGHT].P();forks[LEFT].P();}eating();forks[LEFT].V();forks[RIGHT].V();}
}

在这种策略下就可以有效避免死锁了,并且还可以实现多人同时就餐,不会有第二次尝试中资源的浪费。

利用AND信号量实现

AND信号量我感觉更多是一种思想吧,即在对进程请求的多个资源进行分配时,首先检查这些资源是否都是空闲的,如果的确都是空闲的,则将资源全部分配给该进程,否则一个资源也不分配。很明显,这里检查资源是否空闲过程应该是原子操作才行。我感觉AND信号量具有多种实现的方法啊,比如可以对资源分配的过程加一个互斥访问锁,如下面的代码指示的那样:

Semaphore mutex(1);void philosopher(){while(true){thinking();//I'm hungrymutex.P();forks[LEFT].P();forks[RIGHT].P();mutex.V();eating();forks[LEFT].V();forks[RIGHT].V();}
}

这里由于把对所有资源的分配组织成了一个原子操作,因此也不会出现第一种尝试中的死锁现象。

另一种方法是把哲学家左右的两只叉子抽象为一个资源,即为每个哲学家设置一个信号量。哲学家请求叉子时,首先检查他相邻的两个哲学家是否在就餐,只有相邻两个哲学家都没有获得叉子的时候才为他分配同时分配两只叉子。应该注意到,检查相邻哲学家是否在就餐,其实就是在检查是否左右两只叉子是否都是空闲状态,因此该检查的动作应该被封装为原子操作。为了检查哲学家的状态,为每个哲学家设置三个状态,即思考THINKING,饥饿HUNGRY,和就餐EATING。实现的伪代码如下:

Semaphore phis[5]; 			//one semaphore for each philosopher
Semaphore mutex(1);
int state[5];
for(phi : phis)phi.sem = 0;			//not available in the beginningvoid check(i){if(state[i] == HUNGRY && (state[LEFT] != EATING && state[RIGHT] != EATING)){state[i] = EATING;phis[i].V();	}
}//for philosopher i
void philosopher(){while(true){thinking();//I'm hungrystate[i] = HUNGRYmutex.P();check(i);mutex.V();phis[i].P();eating();state[i] = THINKING;mutex.P();check(LEFT);check(RIGHT);mutex.V();}
}

在 ucore lab7_report 中还叙述了如何通过条件变量和管程,利用AND信号量机制来实现哲学家就餐问题。

这篇关于哲学家就餐问题及其实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Flutter实现文字镂空效果的详细步骤

《Flutter实现文字镂空效果的详细步骤》:本文主要介绍如何使用Flutter实现文字镂空效果,包括创建基础应用结构、实现自定义绘制器、构建UI界面以及实现颜色选择按钮等步骤,并详细解析了混合模... 目录引言实现原理开始实现步骤1:创建基础应用结构步骤2:创建主屏幕步骤3:实现自定义绘制器步骤4:构建U

MySQL主从同步延迟问题的全面解决方案

《MySQL主从同步延迟问题的全面解决方案》MySQL主从同步延迟是分布式数据库系统中的常见问题,会导致从库读取到过期数据,影响业务一致性,下面我将深入分析延迟原因并提供多层次的解决方案,需要的朋友可... 目录一、同步延迟原因深度分析1.1 主从复制原理回顾1.2 延迟产生的关键环节二、实时监控与诊断方案

SpringBoot中四种AOP实战应用场景及代码实现

《SpringBoot中四种AOP实战应用场景及代码实现》面向切面编程(AOP)是Spring框架的核心功能之一,它通过预编译和运行期动态代理实现程序功能的统一维护,在SpringBoot应用中,AO... 目录引言场景一:日志记录与性能监控业务需求实现方案使用示例扩展:MDC实现请求跟踪场景二:权限控制与

Android实现定时任务的几种方式汇总(附源码)

《Android实现定时任务的几种方式汇总(附源码)》在Android应用中,定时任务(ScheduledTask)的需求几乎无处不在:从定时刷新数据、定时备份、定时推送通知,到夜间静默下载、循环执行... 目录一、项目介绍1. 背景与意义二、相关基础知识与系统约束三、方案一:Handler.postDel

SQLyog中DELIMITER执行存储过程时出现前置缩进问题的解决方法

《SQLyog中DELIMITER执行存储过程时出现前置缩进问题的解决方法》在SQLyog中执行存储过程时出现的前置缩进问题,实际上反映了SQLyog对SQL语句解析的一个特殊行为,本文给大家介绍了详... 目录问题根源正确写法示例永久解决方案为什么命令行不受影响?最佳实践建议问题根源SQLyog的语句分

使用Python实现IP地址和端口状态检测与监控

《使用Python实现IP地址和端口状态检测与监控》在网络运维和服务器管理中,IP地址和端口的可用性监控是保障业务连续性的基础需求,本文将带你用Python从零打造一个高可用IP监控系统,感兴趣的小伙... 目录概述:为什么需要IP监控系统使用步骤说明1. 环境准备2. 系统部署3. 核心功能配置系统效果展

解决IDEA报错:编码GBK的不可映射字符问题

《解决IDEA报错:编码GBK的不可映射字符问题》:本文主要介绍解决IDEA报错:编码GBK的不可映射字符问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录IDEA报错:编码GBK的不可映射字符终端软件问题描述原因分析解决方案方法1:将命令改为方法2:右下jav

Python实现微信自动锁定工具

《Python实现微信自动锁定工具》在数字化办公时代,微信已成为职场沟通的重要工具,但临时离开时忘记锁屏可能导致敏感信息泄露,下面我们就来看看如何使用Python打造一个微信自动锁定工具吧... 目录引言:当微信隐私遇到自动化守护效果展示核心功能全景图技术亮点深度解析1. 无操作检测引擎2. 微信路径智能获

Python中pywin32 常用窗口操作的实现

《Python中pywin32常用窗口操作的实现》本文主要介绍了Python中pywin32常用窗口操作的实现,pywin32主要的作用是供Python开发者快速调用WindowsAPI的一个... 目录获取窗口句柄获取最前端窗口句柄获取指定坐标处的窗口根据窗口的完整标题匹配获取句柄根据窗口的类别匹配获取句

MyBatis模糊查询报错:ParserException: not supported.pos 问题解决

《MyBatis模糊查询报错:ParserException:notsupported.pos问题解决》本文主要介绍了MyBatis模糊查询报错:ParserException:notsuppo... 目录问题描述问题根源错误SQL解析逻辑深层原因分析三种解决方案方案一:使用CONCAT函数(推荐)方案二: