C语言实现银行家算法

简介: C语言实现银行家算法

一.银行家算法

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

2.初始化数据显示

3.进行安全算法

4.模拟资源申请

五.实验总结

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

六.补充

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



相关文章
|
9天前
|
搜索推荐 C语言
【排序算法】快速排序升级版--三路快排详解 + 实现(c语言)
本文介绍了快速排序的升级版——三路快排。传统快速排序在处理大量相同元素时效率较低,而三路快排通过将数组分为三部分(小于、等于、大于基准值)来优化这一问题。文章详细讲解了三路快排的实现步骤,并提供了完整的代码示例。
32 4
|
3月前
|
存储 算法 C语言
"揭秘C语言中的王者之树——红黑树:一场数据结构与算法的华丽舞蹈,让你的程序效率飙升,直击性能巅峰!"
【8月更文挑战第20天】红黑树是自平衡二叉查找树,通过旋转和重着色保持平衡,确保高效执行插入、删除和查找操作,时间复杂度为O(log n)。本文介绍红黑树的基本属性、存储结构及其C语言实现。红黑树遵循五项基本规则以保持平衡状态。在C语言中,节点包含数据、颜色、父节点和子节点指针。文章提供了一个示例代码框架,用于创建节点、插入节点并执行必要的修复操作以维护红黑树的特性。
102 1
|
20天前
|
存储 算法 数据管理
C语言算法复杂度
【10月更文挑战第20天】
C语言算法复杂度
|
10天前
|
搜索推荐 算法 C语言
【排序算法】八大排序(上)(c语言实现)(附源码)
本文介绍了四种常见的排序算法:冒泡排序、选择排序、插入排序和希尔排序。通过具体的代码实现和测试数据,详细解释了每种算法的工作原理和性能特点。冒泡排序通过不断交换相邻元素来排序,选择排序通过选择最小元素进行交换,插入排序通过逐步插入元素到已排序部分,而希尔排序则是插入排序的改进版,通过预排序使数据更接近有序,从而提高效率。文章最后总结了这四种算法的空间和时间复杂度,以及它们的稳定性。
51 8
|
10天前
|
搜索推荐 算法 C语言
【排序算法】八大排序(下)(c语言实现)(附源码)
本文继续学习并实现了八大排序算法中的后四种:堆排序、快速排序、归并排序和计数排序。详细介绍了每种排序算法的原理、步骤和代码实现,并通过测试数据展示了它们的性能表现。堆排序利用堆的特性进行排序,快速排序通过递归和多种划分方法实现高效排序,归并排序通过分治法将问题分解后再合并,计数排序则通过统计每个元素的出现次数实现非比较排序。最后,文章还对比了这些排序算法在处理一百万个整形数据时的运行时间,帮助读者了解不同算法的优劣。
39 7
|
1月前
|
存储 算法 C语言
【C语言】二分查找算法
【C语言】二分查找算法
|
1月前
|
搜索推荐 C语言 C++
【C语言】指针篇-精通库中的快速排序算法:巧妙掌握技巧(4/5)
【C语言】指针篇-精通库中的快速排序算法:巧妙掌握技巧(4/5)
|
3月前
|
机器学习/深度学习 存储 并行计算
C语言与机器学习:K-近邻算法实现
C语言与机器学习:K-近邻算法实现
61 0
|
5月前
|
存储 算法 C语言
二分查找算法的概念、原理、效率以及使用C语言循环和数组的简单实现
二分查找算法的概念、原理、效率以及使用C语言循环和数组的简单实现
|
5月前
|
算法 C语言
C语言----判断n是否是2的次方数,利用到按位与&,算法n&(n-1)
C语言----判断n是否是2的次方数,利用到按位与&,算法n&(n-1)