C语言实现银行家算法

news2024/12/25 12:20:14

一.银行家算法

1.由来

银行家算法最初是由荷兰计算机科学家艾兹赫尔·迪杰斯特拉(Edsger W. Dijkstra)于1965年提出的。当时他正致力于解决多道程序设计中产生的死锁问题。在多道程序设计中,由于不同进程之间共享有限的系统资源,如内存、I/O设备等,因此存在一个进程等待其他进程释放资源而导致所有进程都无法执行完毕的情况,称为死锁。为了避免死锁的发生,需要一种能够动态地分配和撤销资源的方法。

银行家算法就是针对这个问题而提出的一种资源分配算法。它基于资源分配图和安全序列的概念,通过动态计算系统当前的安全状态来判断是否可以分配资源,并且仅在分配后不会导致死锁的情况下执行分配。银行家算法成为了解决死锁问题的经典算法之一,在操作系统中得到广泛应用,并且也启发了许多后来的研究工作。

2.介绍

银行家算法是一种用于避免死锁的经典算法,通常应用于操作系统中。该算法通过检查资源分配状态来判断是否可以满足进程的请求,并且仅在满足所有进程请求时才执行分配。以此来保证系统不会陷入死锁。

银行家算法的基本思想是,在计算机系统中,对于每种资源都设置一个最大需求量和当前可用量,当一个进程提出资源请求时,系统首先检查该进程是否满足其最大需求量限制,如果请求合法,则尝试分配资源给该进程并检查是否会导致系统进入不安全状态(即可能发生死锁),如果不会,则分配资源;否则,拒绝该进程的请求。具体实现中,通常使用银行家算法数据结构来记录每个进程的最大需求量、已分配数量和当前需要数量等信息,以及每种资源的总量和可用数量等信息。根据这些信息,可以动态地计算系统当前的安全状态,从而有效地避免死锁的发生。

二.基本原理

1.文字理解

银行家算法的原理基于资源分配图和安全序列

在银行家算法中,每个进程有一个最大需求量向量、一个已分配资源向量和一个当前需要的资源向量。系统也有一个可用资源向量和一个总资源向量。当一个进程请求一定数量的资源时,系统必须确定这个请求是否能被满足,并在不进入死锁状态的前提下将资源分配给该进程。

为了检查是否能够满足该请求并避免死锁,银行家算法实现了安全性检查机制。安全性检查机制会计算出所有未满足进程的最大需求量和当前需要量之和(即还需要的资源),然后尝试找到一个安全序列,如果可以找到安全序列,则说明该请求可以被满足而不会导致死锁。

安全序列是指一个进程执行完毕并释放它所占用的所有资源后,系统能够满足所有其他进程的最大需求量和当前需要量之和的一个序列。如果能够找到安全序列,则说明当前状态是安全的,可以分配资源给请求进程;否则,该请求就不能被满足,应该等待资源。

因此,银行家算法的原理是根据资源分配情况动态地计算系统的安全状态,从而决定是否分配资源,以避免死锁的发生。

2.程序设计基本图

jLfd.jpg

3.安全算法原理

jabW.jpg

三.重要部分功能及实现

1.初始化变量定义

说明:如果我们改变资源种类和进程数量,可以在源代码开始定义变量处改变宏定义p,s的值,后续代
码的p,s的值也随之改变。如果我们采取读取文件的方法,我们可以分别建立一个资源种类和进程的.txt
文档读入,然后统计其数量赋给p,s变量。

/*Author:Cnkizy
数据参考 P121 4.银行家算法之例
*/
#include<stdio.h>
#define Pcount 5 //5个进程
#define Scount 3 //3类资源
int Available[Scount];//可利用资源向量
int Max[Pcount][Scount];//最大需求矩阵 可以通过Need+Allocation算出来
int Allocation[Pcount][Scount];//分配矩阵
int Need[Pcount][Scount];//需求矩阵
//int SouresMax[Scount] = { 10,5,7 };//这里给ABC三类资源的数量为10,5,7
/*资源分配表,必要的一些数据如下
	Max		Allocation	Need	Available
	P0		0 1 0		7 4 3	3 3 2
	P1		2 0 0		1 2 2
	P2		3 0 2		6 0 0
	P3		2 1 1		0 1 1
	P4		0 0 2		4 3 1
*/

2.计算最大需求量

说明:根据题意关系我们可知,最大资源需求量可以有已分配资源量和仍需求资源量求出。这样我们可
以减少读取最大资源需求量的文件操作。

//计算最大需求数量
void CalcMax() {
	for (int i = 0; i < p; i++) {
		for (int j = 0; j < s; j++) {
			Max[i][j] = Need[i][j] + Allocation[i][j];
		}
	}
}

3.初始化数据

说明:这里我们通过三次文件读取操作,分别将存有已分配资源数量、仍需求资源数量、待分配资源数
量分别存入其数组中,并调用计算最大需求量函数计算出最大资源需求量,完成数据初始化。

//初始化数据,资源分配表
void InitializeData() {
	//读取已分配资源
	int b[100];
	FILE* fp;
	fp = fopen("Allocation.txt", "r");
	if (fp == NULL) {
		printf("file is error.");
		return -1;
	}
	for (int j = 0; j < 15; j++) {
		fscanf(fp, "%d", &b[j]);
	}
	fclose(fp);
	int k = 0;
	for (int j = 0; j < p; j++) {
		for (int i = 0; i < s; i++) {
			Allocation[j][i] = b[k];
			k++;
		}
	}
	//读取仍需求资源
	int a[100];
	FILE* fpread;
	fpread = fopen("need.txt", "r");
	if (fpread == NULL) {
		printf("file is error.");
		return -1;
	}
	for (int j = 0; j < 15; j++) {
		fscanf(fpread, "%d", &a[j]);
	}
	fclose(fpread);
	int m = 0;
	for (int j = 0; j < p; j++) {
		for (int i = 0; i < s; i++) {
			Need[j][i] = a[m];
			m++;
		}
	}
	//读取待分配资源
    int c[100];
	FILE* fpr;
	fpr = fopen("Available.txt", "r");
	if (fpr == NULL) {
		printf("file is error.");
		return -1;
	}
	for (int j = 0; j < 15; j++) {
		fscanf(fpr, "%d", &c[j]);
	}
	fclose(fpr);
	int n = 0;
	for (int j = 0; j < p; j++) {
		for (int i = 0; i < s; i++) {
			Available[j][i] = c[n];
			n++;
		}
	}
	CalcMax();
}

4.显示当前资源分配

说明:这里我们建立一个函数来显示当前资源分配情况,方便阅读,以免我们分配资源时没有可视化的数据情况,从而出现错误。

//查看当前资源分配表
void ShowData(int line) {
	printf("	Max	Alloca	Need	Available\n");
	for (int i = 0; i < p; i++) {
		printf("p%d:\t", i);
		for (int j = 0; j < s; j++) {
			printf("%d ", Max[i][j]);
		}
		printf("\t");
		for (int j = 0; j < s; j++) {
			printf("%d ", Allocation[i][j]);
		}
		printf("\t");
		for (int j = 0; j < s; j++) {
			printf("%d ", Need[i][j]);
		}

		if (line == i) {
			printf("\t");
			for (int j = 0; j < s; j++) {
				printf("%d ", Available[j]);
			}
		}

		printf("\n");
	}

}

5.安全型算法

说明:这里写的是银行家算法里面最重要的部分安全性检测算法,首先我们设置两个向量:工作向量Work,表示系统可提供给进程继续运行所需的各类资源数目,它含有m个元素,当执行安全算法开始时,Work=Available;Finish,表示系统是否有足够的资源分配给进程,使之完成运行。开始时先使Finish[i]=false,当有足够的资源分配给进程时,再令Finish[i]=true。

//向量相加 a = a+b
void Add(int* a, int b[s]) {
	for (int i = 0; i < s; i++) {
		a[i] = a[i] + b[i];
	}
}
//向量相减 a = a-b
void Minus(int* a, int b[s]) {
	for (int i = 0; i < s; i++) {
		a[i] = a[i] - b[i];
	}
}

//资源比较   a<=b 返回1     a>b 返回0
int Equals(int a[s], int b[s]) {
	for (int i = 0; i < s; i++) {
		if (a[i] > b[i]) return 0;
	}
	return 1;
}

//检查标志所有都为True,是返回1 不是返回0
int CheckFinish(int Finish[p]) {
	for (int i = 0; i < p; i++) {
		if (Finish[i] == 0) return 0;
	}
	return 1;
}
//安全性算法,当前是否处于安全状态
int CheckSafe() {
	printf("开始安全性检查(输出一个安全序列):\n");
	//步骤1 设置两个向量
	int Finish[p] = { 0 };//是否被处理过,初始值全为False,被检查过才置为True
	int Work[s] = { 0 };//工作向量	
	Add(Work, Available);//首先让Work = Available
	//步骤2 从进程集合寻找符合下列条件的进程
	//Finish[i] =  false;
	//Need[i,j] <= Work[j];
	for (int i = 0; i < p; i++) {
		if (Finish[i])continue;//已经标记为True就跳过
		if (!Equals(Need[i], Work))continue;//Need[i,j] > Work[j] 就跳过。
		//上述条件成立,执行步骤3
		Add(Work, Allocation[i]);//Work += Allocation;
		Finish[i] = 1;//Finish[i]=True;	
		printf("P%d进程,Work=%d %d %d,Finish=true,安全状态\n", i, Work[0], Work[1], Work[2]);
		i = -1;//返回步骤2
	}
	//步骤4 判断Finish
	if (CheckFinish(Finish)) {
		printf("安全状态检查完毕:【Finish全为true,系统处于安全状态】\n");
		return 1;//全为True		
	}
	printf("安全状态检查完毕:【Finish存在False,系统处于不安全状态】\n");
	return 0;//存在False
}

6.资源请求

说明:这里我们写的是申请资源函数,对于某个进程的资源申请,系统首先检测申请资源的进程要求是否合理,如果不合理会驳回要求,如果合理系统会先模拟把该进程申请的资源分配给该资源,然后对这种情况进行安全算法检测,如果存在一个安全序列,系统则会满足该进程的要求,然后显示当前资源分配情况。

//带命令提示符提示的请求
void RequestShowMsg(int P, int R[s]) {
	//进程P 申请资源Request{1,0,2}
	printf("\n模拟分配资源:P%d申请资源 %d %d %d\n======================\n",P, R[0], R[1], R[2]);
	int State = Apply(P, R);
	if (State) {
		printf("本次资源分配成功!\n");
		ShowData(0);
	}else {
		printf("本次资源分配失败!进程P%d需要等待\n",P);
	}
}
//进程资源请求函数, P:进程i, r申请资源数{1,1,1}     返回1成功 0失败
int Apply(int P, int Request[s]) {
	printf("进程P%d申请资源%d %d %d:\n", P, Request[0], Request[1], Request[2]);
	//步骤1 进行资源检查Request <= Need才能执行步骤2
	if (!Equals(Request, Need[P])) {
		printf("进程P%d,Request:%d %d %d > Need:%d %d %d 申请失败,所需资源数超过宣布最大值!\n", P, Request[0], Request[1], Request[2], Need[P][0], Need[P][1], Need[P][2]);
		return 0;
	}
	//步骤2 进行资源检查Request <= Available才能执行步骤3
	if (!Equals(Request, Available)) {
		printf("进程P%d,Request:%d %d %d > Available:%d %d %d 申请失败,尚无足够资源,该进程需要等待!\n", P, Request[0], Request[1], Request[2], Available[0], Available[1], Available[2]);
		return 0;
	}
	printf("进程P%d,Request:%d %d %d <= Need:%d %d %d\n", P, Request[0], Request[1], Request[2], Need[P][0], Need[P][1], Need[P][2]);
	printf("进程P%d,Request:%d %d %d <= Available:%d %d %d \n", P, Request[0], Request[1], Request[2], Available[0], Available[1], Available[2]);
	//步骤3 试分配资源给进程P
	Minus(Available, Request);//Available -= Request
	Add(Allocation[P], Request); //Allocation += Request
	Minus(Need[P], Request);//Need -= Request
	//步骤4 安全性检查
	int Safestate = CheckSafe();
	if (Safestate) {
		return Safestate;//分配后处于安全状态 分配成功
	}
	//分配后处于不安全状态 分配失败,本次分配作废,回复原来的资源分配状态
	Add(Available, Request);         //Available += Request
	Minus(Allocation[P], Request);   //Allocation -= Request
	Add(Need[P], Request);          //Need += Request
	return Safestate;
}

四.各部分实现结果

1.文件内容展示

jbMi.jpg

2.初始化数据显示

jcEp.jpg

3.进行安全算法

jhOf.jpg

4.模拟资源申请

jpnZ.jpg

五.实验总结

多个进程同时运行时,系统根据各类系统资源的最大需求和各类系统的剩余资源为进程安排安全序列,使得系统能快速且安全地运行进程,不至发生死锁。银行家算法是避免死锁的主要方法,其思路在很多方面都非常值得我们来学习借鉴。

六.补充

本次博客可以看作是上一篇博客关于避免死锁的补充。多线程

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

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

相关文章

【JavaEE初阶】多线程(一)认识线程 线程的创建 Thread的用法

摄影分享&#xff01; 文章目录 认识线程&#xff08;Thread&#xff09;概念执行多线程编程创建线程的写法1.继承Thread&#xff0c;重写run2.实现Runnable接口3.使用匿名内部类&#xff0c;继承Thread4.使用匿名内部类&#xff0c;实现Runable5.使用Lambda表达式 Thread用法…

C语言模拟银行排队叫号(链队)

一.队列 队列是一种具有先进先出&#xff08;FIFO&#xff09;特性的线性数据结构&#xff0c;它只允许在队列的两端进行插入和删除操作。队列的一端称为队尾&#xff08;rear&#xff09;&#xff0c;另一端称为队头&#xff08;front&#xff09;。新元素总是插入在队列的队…

怎么把m4a转换成mp3,分享几个方法给大家!

录音文件中经常出现m4a后缀的音频格式&#xff0c;但通常只能在特定的音频播放器中播放。如果你想把m4a转换成mp3&#xff0c;下面是四种简单易行的方法&#xff0c;适用于Windows 10操作系统。 方法一&#xff1a;使用记灵在线工具转换m4a成mp3 工具地址&#xff1a;记灵在线…

elsticsearch入门

查看所有索引&#xff08;表&#xff09; 向索引&#xff08;表&#xff09;中添加数据&#xff1a; 自定义id添加数据&#xff1a; 自定义id添加数据&#xff1a;方式二 查询数据&#xff1a; 查询索引&#xff08;表&#xff09;中全部数据&#xff1a; 全量修改单条数据&…

File类与IO流

1. java.io.File类的使用 1.1 概述 File类及本章下的各种流&#xff0c;都定义在java.io包下。一个File对象代表硬盘或网络中可能存在的一个文件或者文件目录&#xff08;俗称文件夹&#xff09;&#xff0c;与平台无关。&#xff08;体会万事万物皆对象&#xff09;File 能新…

【Leetcode -剑指Offer 22.链表中倒数第k个结点 -203.移除链表元素】

Leetcode Leetcode -剑指Offer 22.链表中倒数第k个结点Leetcode -203.移除链表元素 Leetcode -剑指Offer 22.链表中倒数第k个结点 题目&#xff1a;输入一个链表&#xff0c;输出该链表中倒数第k个节点。为了符合大多数人的习惯&#xff0c;本题从1开始计数&#xff0c;即链表…

数据结构(三)—— 哈希表

文章目录 一、哈希表积累1.1 哈希map1.2 哈希set 二、哈希表基础三、题3.1 242 有效的字母异位词3.2 349 两个数组的交集3.3 202 快乐数3.4 1 两数之和3.5 54 四数相加II 一、哈希表积累 什么时候想到用哈希法&#xff1a;当要需要查询一个元素是否出现过、判断一个元素是否出…

awvs安装批量扫描

文章目录 安装批量扫描 安装 1.2.AWVS下载   该工具可在官方网站下载&#xff0c;但免费下载的是14天试用版本。   官网下载&#xff1a;AWVS   百度云下载&#xff1a;百度云下载https://pan.baidu.com/s/1UO7GzL0CMemJ_TMQnHNOuA?pwdg1bm 提取码&#xff1a;g1bm 批…

浏览器状态同步和路由-前端路由和服务端路由原理

目录 前端路由和History API 浏览文境&#xff08;Browser Context&#xff09; 会话历史&#xff08;Session History&#xff09; History API history.go()切换当前会话&#xff0c;并不改变会话栈 history.back() history.go(-1) & history.forward() history.…

5.4、服务器编程基本框架和两种高效的事件处理模式

5.4、服务器编程基本框架和两种高效的事件处理模式 1.服务器编程基本框架2.两种高效的事件处理模式①Reactor模式②Proactor模式③模拟Proactor模式 1.服务器编程基本框架 模块功能I/O 处理单元处理客户连接&#xff0c;读写网络数据逻辑单元业务进程或线程网络存储单元数据库、…

18.网络爬虫—Scrapy实战演示

网络爬虫—Scrapy实战演示 Scrapy Shell简介进入shell调试网站启动Scrapy Shell 查看目标网站获取网站源代码常用方法调试xpath提取数据Scrapy请求子页面请求及返回处理创建项目创建爬虫 数据解析写入csv文件后记 前言&#xff1a; &#x1f3d8;️&#x1f3d8;️个人简介&…

Java,jdbc,jvm

1、数据删除 物理删除 直接发送delete语句 就是物理删除 这种删除 删除之后不可恢复逻辑删除 本质是更新 0 表示不可用 1 可用状态 update emp set is_active where id ?2、数据更新 1 显示所有数据 2. 点击修改按钮 此时 应该执行的动作-查询该用户信息 目的是将当前用户…

并发编程的那些事

目录 一、并发编程的目的 二、线程和进程2.1 什么是线程2.2 进程2.3 一个普通Java 程序包含哪些线程 三、并发、并行四、线程的六个状态五、wait 和sleep的区别5.1 位于不同的类5.2 关于锁的释放 一、并发编程的目的 并发编程的目的是为了让程序运行得更快&#xff0c;但是&…

类ChatGPT逐行代码解读(1/2):从零起步实现Transformer、ChatGLM-6B

前言 最近一直在做类ChatGPT项目的部署 微调&#xff0c;关注比较多的是两个&#xff1a;一个LLaMA&#xff0c;一个ChatGLM&#xff0c;会发现有不少模型是基于这两个模型去做微调的&#xff0c;说到微调&#xff0c;那具体怎么微调呢&#xff0c;因此又详细了解了一下微调代…

ggplot中坐标轴和图例的相关处理

文章目录 改变坐标轴和图例的名称方法1, labs()方法2&#xff0c;scale_xxx_discrete/continuous() 删除坐标轴和图例的名称方法1&#xff0c; labs()方法2&#xff0c;scale_xxx_discrete/continuous()方法3&#xff0c;theme()方法4&#xff0c;guides()可以去图例名称 改变图…

osgwidget 使用 方法以及案例分享

osgwidget 使用 方法以及案例分享 按钮 一个常见的 osg::Widget 就是按钮。下面的代码展示了如何使用 osg::Switch 和 osgText 创建一个简单的按钮&#xff1a; osg::ref_ptr<osg::Switch> buttonSwitch new osg::Switch(); osg::ref_ptr<osgText::Text> buttonTe…

[论文阅读] (29)李沐老师视频学习——2.研究的艺术·找问题和明白问题的重要性

《娜璋带你读论文》系列主要是督促自己阅读优秀论文及听取学术讲座&#xff0c;并分享给大家&#xff0c;希望您喜欢。由于作者的英文水平和学术能力不高&#xff0c;需要不断提升&#xff0c;所以还请大家批评指正&#xff0c;非常欢迎大家给我留言评论&#xff0c;学术路上期…

写在28岁,回看3年前的自己,庆幸当时入了软件测试这行

为什么会学习软件测试&#xff1f; 已经28岁了&#xff0c;算一下快过去3年了&#xff0c;刚毕业那会工作了一年&#xff0c;因为自己当时很迷茫&#xff08;觉得自己挺废的&#xff09;&#xff0c;所以就没去工作就一直在家&#xff0c;家里固定每个月给点生活费&#xff0c…

人工智能、ChatGPT等火爆的当下 AI大模型爆发

4月18日&#xff0c;火山引擎在其举办的“原动力大会”上发布自研DPU等系列云产品&#xff0c;并推出新版机器学习平台&#xff1a;支持万卡级大模型训练、微秒级延迟网络&#xff0c;让大模型训练更稳更快。火山引擎总裁谭待表示&#xff0c;AI大模型有巨大潜力和创新空间&…

chatgpt实际是怎样工作的?

文章翻译自&#xff1a; https://www.assemblyai.com/blog/how-chatgpt-actually-works/ ChatGPT 是 OpenAI 的最新语言模型&#xff0c;比其前身 GPT-3 有了重大改进。与许多大型语言模型类似&#xff0c;ChatGPT 能够为不同目的生成多种样式的文本&#xff0c;但具有更高的精…