哲学家就餐问题及其实现

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

相关文章

MySQL中查找重复值的实现

《MySQL中查找重复值的实现》查找重复值是一项常见需求,比如在数据清理、数据分析、数据质量检查等场景下,我们常常需要找出表中某列或多列的重复值,具有一定的参考价值,感兴趣的可以了解一下... 目录技术背景实现步骤方法一:使用GROUP BY和HAVING子句方法二:仅返回重复值方法三:返回完整记录方法四:

IDEA中新建/切换Git分支的实现步骤

《IDEA中新建/切换Git分支的实现步骤》本文主要介绍了IDEA中新建/切换Git分支的实现步骤,通过菜单创建新分支并选择是否切换,创建后在Git详情或右键Checkout中切换分支,感兴趣的可以了... 前提:项目已被Git托管1、点击上方栏Git->NewBrancjsh...2、输入新的分支的

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Python实现对阿里云OSS对象存储的操作详解

《Python实现对阿里云OSS对象存储的操作详解》这篇文章主要为大家详细介绍了Python实现对阿里云OSS对象存储的操作相关知识,包括连接,上传,下载,列举等功能,感兴趣的小伙伴可以了解下... 目录一、直接使用代码二、详细使用1. 环境准备2. 初始化配置3. bucket配置创建4. 文件上传到os

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

关于集合与数组转换实现方法

《关于集合与数组转换实现方法》:本文主要介绍关于集合与数组转换实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、Arrays.asList()1.1、方法作用1.2、内部实现1.3、修改元素的影响1.4、注意事项2、list.toArray()2.1、方

使用Python实现可恢复式多线程下载器

《使用Python实现可恢复式多线程下载器》在数字时代,大文件下载已成为日常操作,本文将手把手教你用Python打造专业级下载器,实现断点续传,多线程加速,速度限制等功能,感兴趣的小伙伴可以了解下... 目录一、智能续传:从崩溃边缘抢救进度二、多线程加速:榨干网络带宽三、速度控制:做网络的好邻居四、终端交互

java实现docker镜像上传到harbor仓库的方式

《java实现docker镜像上传到harbor仓库的方式》:本文主要介绍java实现docker镜像上传到harbor仓库的方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 前 言2. 编写工具类2.1 引入依赖包2.2 使用当前服务器的docker环境推送镜像2.2

Redis出现中文乱码的问题及解决

《Redis出现中文乱码的问题及解决》:本文主要介绍Redis出现中文乱码的问题及解决,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1. 问题的产生2China编程. 问题的解决redihttp://www.chinasem.cns数据进制问题的解决中文乱码问题解决总结

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的