C语言实现银行家算法

这篇具有很好参考价值的文章主要介绍了C语言实现银行家算法。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一.银行家算法

1.由来

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

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

2.介绍

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

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

二.基本原理

1.文字理解

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

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

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

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

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

2.程序设计基本图

银行家算法c语言实现,考研,# 操作系统,算法,c语言,网络

3.安全算法原理

银行家算法c语言实现,考研,# 操作系统,算法,c语言,网络

三.重要部分功能及实现

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.文件内容展示

银行家算法c语言实现,考研,# 操作系统,算法,c语言,网络

2.初始化数据显示

银行家算法c语言实现,考研,# 操作系统,算法,c语言,网络

3.进行安全算法

银行家算法c语言实现,考研,# 操作系统,算法,c语言,网络

4.模拟资源申请

银行家算法c语言实现,考研,# 操作系统,算法,c语言,网络

五.实验总结

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

六.补充

本次博客可以看作是上一篇博客关于避免死锁的补充。多线程文章来源地址https://www.toymoban.com/news/detail-744731.html

到了这里,关于C语言实现银行家算法的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 操作系统--银行家算法

    目录 1.产生死锁的原因及必要条件   1.1产生死锁的原因   1.2产生死锁的必要条件 2.处理死锁的方法 3.银行家算法 4.安全性算法 5.算法实现        如果一个进程集合里面的每个进程都在等待这个集合中的其他一个进程(包括自身)才能继续往下执行,若无外力他们将无法推

    2024年02月01日
    浏览(32)
  • 操作系统实验——银行家算法

    掌握银行家算法思想,并能编程实现。 1、在Linux环境下编译运行程序; 2、按照教材的算法编写; 3、(*)输入数据从文本文件中读出,不从键盘录入,数据文件格式见以下说明; 4、主要数据结构的变量名和教材中的一致,包括Available、Max、Allocation、Need、Request、Work、Fin

    2024年02月01日
    浏览(39)
  • 操作系统实验 银行家算法C++

    实验目的: 编程实现安全性算法及银行家算法,以帮助深刻理解银行家算法避免死锁的原理。 算法流程图:     实现代码:    验证数据: 运行结果:     说明: 本文章是在原作者的银行家算法文章基础上依据实验课要求修改和完善的,仅供参考,侵权删。  原作者地址

    2024年02月05日
    浏览(44)
  • 【操作系统】银行家算法个人出题例题 (含答案)

    1.银行家算法是代表性的避免死锁的算法,在进程调度中具有重要作用。请结合所学知识回答以下问题:(23分——加长版) (1)银行家算法使用的四个必要的数据结构是:可用资源向量Available,____________,分配矩阵Allocation,需求矩阵Need。(1分) (2)以下是银行家算法具体实现

    2024年02月12日
    浏览(35)
  • 操作系统实验二死锁避免之银行家算法的模拟

    死锁  (1)定义  (2)死锁产生的原因  (3)死锁产生的必要条件  (4)死锁的处理策略 银行家算法  (1)核心思想  (2)数据结构  (3)算法描述    (4)  安全性检查算法 银行家算法的模拟 (1)数据结构 (2)完整代码 (3)测试 所谓死锁,是指多个进程因为竞争资

    2024年02月01日
    浏览(64)
  • 银行家算法(C++实现)

    目录 一、银行家算法概述 二、银行家算法需要的数组结构 三、算法概述 1.安全性算法 2.银行家算法 四、代码实现 五、实验结果验证 银行家算法 (Banker\\\'s Algorithm)是一个避免死锁(Deadlock)的著名算法,是由艾兹格·迪杰斯特拉在1965年为T.H.E系统设计的一种避免死锁产生的

    2024年01月24日
    浏览(30)
  • 银行家算法——C++实现 [ 开源代码 + 详细解析 ]

    ✅ (原创,纯手敲,开源免费,2021的最后一篇) Banker Algorithm 🏦 ◆ 说明 :上述算法的核心实现采用了 “DFS + 回溯” 的方法,详见后文的源代码。另外,如果把 C++ 代码里面的 “ p_num=1; ” 注释掉,得到的是另一个结果。我虽然输入是“0”,但代码里后面我直接把 p_num 赋值

    2023年04月26日
    浏览(74)
  • 银行家算法--申请资源

    问题描述: 输入N个进程(N=100),以及M类资源(M=100),初始化各种资源的总数,T0时刻资源的分配情况。例如: 假定系统中有5个进程{P0,P1,P2,P3,P4}和三类资源{A,B,C},各种资源的数量分别为10、5、7,在T0时刻的资源分配图如下: 输入申请资源的进程以及申请各类资源的数

    2024年02月03日
    浏览(59)
  • 银行家算法

    银行家算法(Banker’s Algorithm)是一个避免死锁(Deadlock)的著名算法,是由艾兹格·迪杰斯特拉在1965年为T.H.E系统设计的一种避免死锁产生的算法。它以银行借贷系统的分配策略为基础,判断并保证系统的安全运行。 假设客户A、B、C 三人向银行贷款用于建房,分别需要贷款

    2024年02月02日
    浏览(37)
  • 银行家算法的实验报告

    一、实验内容 银行家算法是避免死锁的一种重要方法,本实验要求编写和调试一个简单的银行家算法程序。 1.设计进程对各类资源最大申请表示及初值的确定。 2.设定系统提供资源的初始状况。 3.设定每次某个进程对各类资源的申请表示。 4.编制程序,依据银行家算法

    2023年04月26日
    浏览(45)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包