一.银行家算法

1.由来

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

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

2.介绍

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

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

二.基本原理

1.文字理解

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

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

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

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

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

2.程序设计基本图

3.安全算法原理

三.重要部分功能及实现

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    AvailableP0     0 1 0       7 4 3   3 3 2P1     2 0 0       1 2 2P2     3 0 2       6 0 0P3     2 1 1       0 1 1P4     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,被检查过才置为Trueint 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] 就跳过。//上述条件成立,执行步骤3Add(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 判断Finishif (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才能执行步骤2if (!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才能执行步骤3if (!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 试分配资源给进程PMinus(Available, Request);//Available -= RequestAdd(Allocation[P], Request); //Allocation += RequestMinus(Need[P], Request);//Need -= Request//步骤4 安全性检查int Safestate = CheckSafe();if (Safestate) {return Safestate;//分配后处于安全状态 分配成功}//分配后处于不安全状态 分配失败,本次分配作废,回复原来的资源分配状态Add(Available, Request);         //Available += RequestMinus(Allocation[P], Request);   //Allocation -= RequestAdd(Need[P], Request);          //Need += Requestreturn Safestate;
}

四.各部分实现结果

1.文件内容展示

2.初始化数据显示

3.进行安全算法

4.模拟资源申请

五.实验总结

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

六.补充

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

C语言实现银行家算法相关推荐

  1. c语言银行算法管理,C语言实现银行家算法

    C语言实现银行家算法 C语言实现银行家算法(源码.运行结果) 一. 源码 /* 银行家算法 10-22 */ #include //函数声明 void init(int * pcount,int * ...

  2. 银行家算法课程设计java语言_Java语言的银行家算法

    <Java语言的银行家算法>由会员分享,可在线阅读,更多相关<Java语言的银行家算法(12页珍藏版)>请在人人文库网上搜索. 1.湖北中医学院信息技术系 操作系统课程设计操作 ...

  3. 用c语言编写银行家算法,C语言实现银行家算法

    <C语言实现银行家算法>由会员分享,可在线阅读,更多相关<C语言实现银行家算法(8页珍藏版)>请在人人文库网上搜索. 1.C语言实现银行家算法(源码.运行结果)一. 源码/* ...

  4. 银行家算法实验报告c语言版,银行家算法实验报告C语言版.doc

    <操作系统>课程综合性实验报告 姓名: 学号: 2016 年 11 月 20 日 实验题目进程调度算法程序设计一.实验目的 通过对安全性算法和银行家算法的模拟,进一步理解资源分配的基本概念 ...

  5. python银行家算法_Linux 死锁概念与银行家算法python 实现

    一 . 死锁的概念 接上篇 http://shaobaobaoer.cn/archives/680/linux-process-manager-note 在之前的哲学家吃饭的问题中,当每个哲学家都想进 ...

  6. 用C语言或C++编写一个简单的银行家算法模拟程序

    1.问题描述 银行家算法是操作系统中避免死锁的典型算法.用C语言或C++编写一个简单的银行家算法模拟程序,实现多个进程争用系统临界资源时的分配过程.要求程序实现: 1.当一进程请求一组资源时,先确定是 ...

  7. 银行家算法c语言博客,操作系统之银行家算法

    银行家算法是一种在多道程序系统中避免多个进程并发执行所带来的死锁问题.所谓死锁(Deadlock),是指多个进程在运行过程中因争夺资源而造成的一种僵局(DeadlyEmbrace),当进程处于这种状态 ...

  8. 操作系统中避免死锁的银行家算法【表面C++实际C语言】一学就废的菜鸡代码

    文章目录 银行家算法 实验原理 数据结构 初始化 输出资源分配量 安全性算法 银行家算法 完整代码 测试数据 测试结果 第一题 第二题 银行家算法 银行家算法是一种最有代表性的避免死锁的算法.在避免死 ...

  9. c语言银行家算法模拟程序,C语言实现 操作系统 银行家算法

    /**************************************************** 银行家算法 算法思想: 1. 在多个进程中,挑选资源需求最小的进程Pmin, 可能存在多类资 ...

最新文章

  1. NVIDIA深度学习Tensor Core性能解析(上)
  2. border-raduis 在IE8中的兼容性问题
  3. c4d完全学习手册_动态视觉设计就业班,全商业项目实训,一线制作团队10人小班授课,持续提升学习...
  4. php中对象怎么访问i属性_PHP--序列化与反序列化详解
  5. CentOS 7.2修改网卡名称
  6. 如何设计高效测试用例_高效的企业测试-单元和用例测试(2/6)
  7. java基本特性_Java面试总结之Java基础
  8. 图像处理 --- 一、认识图像处理
  9. 2021住房消费品质服务报告
  10. mysql hsqldb_HSQLDB的使用方法
  11. iPhone5/iPad4越狱后10.3.3降级8.4.1详细教程
  12. linux系统的日历如何改,linux下实现农历的日历
  13. php怎么解析josn数据,用PHP解析JSON数据
  14. 程序员自己的商业模式
  15. APISpace 文本情感倾向分析API
  16. Modeling and Enhancing Low-Quality Retinal Fundus Images
  17. OpenStack之八: network服务(端口9696)
  18. 浅谈智能家居安防监控的重要性
  19. 手机连接电脑热点 显示 连接超时,已解决
  20. 微信公众号 Cookie

热门文章

  1. Vertx之Junit单元测试
  2. 三维向量的简单运算(点积、叉积及点到直线的距离)
  3. 7-4是否同一棵二叉搜索树(不建树)
  4. html图片变色属性,用CSS新属性实现特殊的图片显示效果
  5. 2020年浙江高考考python吗_2020年浙江高考使用全国几卷
  6. IN-CONTEXT REINFORCEMENT LEARNING WITH ALGORITHM DISTILLATION翻译
  7. Python_sobel边缘检测
  8. 人行,中行,工行,人行,中行,工行,建行,农行,交行笔试资料合集(坚决保留)建行,农行,交行笔试资料合集(坚决保留)
  9. html开发移动APP窗口大小自适应,移动端web开发,怎么自适应屏幕的大小
  10. enclosing type java_java中no enclosing instance of type * is accessible的解决方法