银行家算法毕业论文.doc
银行家算法毕业论文题目:银行家算法设计摘 要Dijkstra的银行家算法是最有代表性的避免死锁的算法,该算法由于能用于银行系统现金贷款的发放而得名。银行家算法是在确保当前系统安全的前提下推进的。对进程请求先进行安全性检查,来决定资源分配与否,从而确保系统的安全,有效的避免了死锁的发生。该论文在理解和分析了银行家算法的核心思想以及状态的本质涵义的前提下,对算法的实现在总体上进行了设计,包括在对算法分模块设计,并对各个模块的算法思想通过流程图表示,分块编写代码,并进行测试,最后进行程序的测试,在设计思路上严格按照软件工程的思想执行,确保了设计和实现的可行,可信。代码实现采用C语言。 关键词:银行家算法;死锁;避免死锁;安全性序列目 录中文摘要.I1 绪论.1 1.1 课题背景.1 1.2 课题意义.1 1.3 银行家算法.1 1.4 死锁.2 1.5 安全性序列.22 需求分析.3 2.1 问题描述. 3 2.2 基本要求. 3 2.3 数据流模型. 33 概要设计.4 3.1 模块的划分.4 3.2 模块调用关系.4 3.3 各模块之间的接口.4 3.4 程序流程图.54详细设计.6 4.1数据结构选取分析.6 4.2数据结构设计.6 4.3算法整体设计与调用.6 4.4模块设计与时间复杂度分析.7 4.4.1系统资源初始化函数Init_process .7 4.4.2安全性算法Safety_Algorithm .7 4.4.3接受进程请求试分配Attempt_Allocation; .7 4.4.4 对试分配后的系统进行安全性检查Safety_Algorithm.8 4.5程序流图.8 4.5.1系统以及进程资源初始化Init_process的程序流程图.8 4.5.2安全性算法Safety_Algorithm的程序流程图.9 4.5.3接受进程请求试分配Attempt_Allocation的程序流程图.9 4.5.4对试分配后的系统进行安全性检查Safety_Algorithm的程序流程图.9 5 程序分析测试.10 5.1分模块分析与测试.10 5.1.1初始化系统资源模块Init_process的测试10 5.1.2试分配模块Attempt_Allocation的测试11 5.1.3安全模块Safety_Algorithm的调试.11 5.2集成测试.126 小结.13参考文献.14附录(源代码).151 绪论1.1课题背景在多道程序系统中,虽可以借助多个进程的并发执行来改善系统的资源利用率,提高系统吞吐量,但可能发生一种危险死锁,即多个进程在运行过程中因争夺资源而造成的一种僵局,若无外力作用,将无法再向前推进。如此,寻求一种避免死锁的方法便显得有为重要。死锁的产生一般的原因有两点:竞争资源和进程间推进顺序非法。因此,我们只需在当前的有限资源下,找到一组合法的执行顺序,便能很好的避免死锁,我们称它为安全序列。而银行家算法起源于银行系统的发放贷款,和计算机操作系统的资源分配完全符合,因此可以借鉴该算法的思想,设计出一种有效的算法程序,解决该问题。1.2 课题意义(1)运用软件工程的方法指导设计和实现,即是对这学期刚刚学过的软件工程课的复习,又是一次实战演练,从而提高自己的分析问题,解决问题和动手能力;(2)通过整个算法的设计与实现进一步加深了对算法的理解和多道程序下的计算机系统资源分配现状,为以后进一步的学习打下了良好的基础。1.3 银行家算法我们可以把操作系统看作是银行家,操作系统管理的资源相当于银行家管理的资金,进程向操作系统请求分配资源相当于用户向银行家贷款。 为保证资金的安全,银行家规定: (1) 当一个顾客对资金的最大需求量不超过银行家现有的资金时就可接纳该顾客;(2) 顾客可以分歧贷款,但贷款的总数不能超过最大需求量; (3) 当银行家现有的资金不能满足顾客尚需的贷款数额时,对顾客的贷款可推迟支付,但总能使顾客在有限的时间里得到贷款; (4) 当顾客得到所需的全部资金后,一定能在有限的时间里归还所有的资金. 操作系统按照银行家制定的规则为进程分配资源,当进程首次申请资源时,要测试该进程对资源的最大需求量,如果系统现存的资源可以满足它的最大需求量则按当前的申请量分配资源,否则就推迟分配。当进程在执行中继续申请资源时,先测试该进程已占用的资源数与本次申请的资源数之和是否超过了该进程对资源的最大需求量。若超过则拒绝分配资源,若没有超过则再测试系统现存的资源能否满足该进程尚需的最大资源量,若能满足则按当前的申请量分配资源,否则也要推迟分配。1.4 死锁死锁是进程死锁的简称,是由Dijkstra于1965年研究银行家算法时首先提出来的。是指多个进程循环等待它方占有的资源而无限期地僵持下去的局面。很显然,如果没有外力的作用,那麽死锁涉及到的各个进程都将永远处于封锁状态。它是计算机操作系统乃至并发程序设计中最难处理的问题之一。实际上,死锁问题不仅在计算机系统中存在,在我们日常生活中它也广泛存在。在计算机系统中,涉及软件,硬件资源都可能发生死锁。例如:系统中只有一台CD-ROM驱动器和一台打印机,某一个进程占有了CD-ROM驱动器,又申请打印机;另一进程占有了打印机,还申请CD-ROM。结果,两个进程都被阻塞,永远也不能自行解除。1.5安全性序列 安全序列的的实际意义在于:系统每次进行资源分配后,如果对于系统中新的资源状况,存在一个安全序列,则至少存在一条确保系统不会进入死锁的路径。按照该序列,银行家可以实施一个有效的分配过程使得所有客户得到满足 银行家算法的核心在于安全序列的产生。安全序列正是一种安全的进程推进顺序2 需求分析2.1 问题描述运用银行家算法,避免死锁的发生。在确保当前系统安全的前提下推进的。对进程请求先进行安全性检查,来决定资源分配与否,从而确保系统的安全,有效的避免了死锁的发生。问题的关键在于安全性算法,即找安全性序列。2.2 基本要求 (1)从键盘输入当前系统的资源信息,包括当前可用资源,每个进程对各类资源的最大需求量,每个进程当前已分配的各个资源量和每个进程尚需要的各个资源量,输出结果显示在DOS界面上; (2)输入进程请求,按照设计好的安全性算法进行检查,得到结果并输出整个执行过程的相关信息和最终结果(主要包括资源分配表和安全序列) (3)要求要有各种异常的处理,程序的可控制性和可连续性执行。包括对进程的存在有无检查,请求向量的不合法检查,试分配失败后的数据恢复和重新接受进程请求等。 初始化2.3 数据流模型输出结果安全性检查试分配安全性检查输入信息 3 概要设计3.1 模块的划分 由于该算法规模较小,所以选用结构化的设计方法,将该系统划为五块,分别是:(1)主模块,处在整个系统的最高层,负责组织调用其他模块。 (2)初始化模块,负责从键盘读入系统资源和进程状态,并将系统初识资源分配状态打印。 (3)试分配模块,负责处理进程请求,和相应的数据结构的修改,已经特殊情况的处理。 (4)安全性检查,负责试分配后的安全性检查,以及系统不安全时的资源恢复。3.2 模块调用关系各模块之间的调用如下图示:主模块 Main() 初始化 Init_process()试分配 Attempt_Allocation()安全性检查 Safety_Algorithm() 3.3 各模块之间的接口(1) Flag1:试分配模块Attempt_Allocation与安全性检查Safety_Algorithm之间接口Attempt_Allocation通过检查 flag的真假了判断是否执行。(2) pro:一个地址, Safety_Algorithm返回给主模块main的信息,不为NULL时表示试分配成功,否则系统转入相应异常处理。(3) Flag2:Safety_Algorithm与主模块之间的接口,为真则调用打印函数,输出最终结果,否则调用恢复函数,恢复之前系统状态3.4程序流程图4详细设计4.1数据结构选取分析该算法中用到了较多的数据,基于程序的易实现和较好的结构,决定采用结构链表,以进程为单位(结点)。4.2数据结构设计typedef struct my_process int num; /进程标号 int MaxM;/ 表示某个进程对某类资源的最大需求 int AllocationM;/ 表示某个进程已分配到某类资源的个数 int NeedM;/ 表示某个进程尚需要某类资源的个数 struct my_process* next;/指向下一个结点(进程)process; int AvailableM=0;/ 其中每一个数组元素表示当前某类资源的可用数目,初始化为系统所提供的资源的最大数目int RequestM=0; /请求向量int Record_workNM=0;/存储当前work的值,以便输出int SafetyN=0; /存储安全序列,以便后面排序4.3算法整体设计与调用 主函数void main()主要分四大块:(1)首先需要初始化Init_process(process *head,int m,int* count),存储系统当前状态信息;(2)调用安全算法Safety_Algorithm,检测当前系统安全状态,若安全则进行下一步,否则打印相关信息,程序退出。(3) 调用试分配函数Attempt_Allocation,进行试分配,若试分配成功,修改相关数据结构,打印当前系统资源分布图,转下一步,否则,打印提示信息,接收其他请求向量。 (4) 再次调用安全性算法,检查试分配以后的系统安全性,若安全打印安全性序列和当前系统资源分布图,并进入新一轮的执行。否则之前的试分配作废,恢复试分配之前的数据结构,输出相关提示信息,接收下一个进程请求 4.4模块设计与时间复杂度分析 4.4.1系统资源初始化函数 Init_process (1)首先读入系统可用资源。(2)用malloc()函数动态创建进程并且输入相关资源MaxM,AllocationM,NeedM。进程之间以链表组织,输入-1结束初始化。 for(i=0;i<m;i+)scanf("%d",&node.Needi); /初始化资源Insret_Tail(head,node);/插入链尾(*count)+;/资源个数加14.4.2安全性算法Safety_Algorithm 找安全序列,其中用到查找当前合法进程的函数process* Reasonable()(1)先初始化相关数据结构work=(int*)malloc(m*sizeof(int); /当前系统可用资源finish=(int*)malloc(n*sizeof(int); /标记向量,初始为false,通过安全检查后最为true (2)Reasonable(head,finish,work,m,n)函数,找到当前可执行的进程,执行Alter_Data()修改当前数据结构。 (3)while(count<n) 重复步骤(2) 4.4.3接受进程请求,试分配Attempt_Allocation(head,Request,Available,M); (1)先判断标志flag的值,若为真,则执行Attempt_Allocation函数,该 flag是由Safety_Algorithm返回的。(2)进入Safety_Algorithm函数,首先输入进程号,并检查该进程是否存在,存在则输入它的请求向量,并进行检查。requesti<=p->Needirequesti<=availi若以上两个条件都为真,试分配成功,则执行Alter_Data()函数,修改相关数据结构,否则该进程被阻塞,系统转而接受其他请求 availi=availi - requesti;p->Allocationi=p->Allocationi + requesti;p->Needi=p->Needi - requesti;将结果打印出来。4.4.4 对试分配后的系统,进行安全性检查Safety_Algorithm。执行过程同4.4.2大致一样,唯一一点在于当找不到安全序列时,将本次试分配作废,恢复该次试分配之前的数据结构。4.5程序流图4.5.1系统以及进程资源初始化Init_process的程序流程图,如下图4-1第一步,读入当前系统可用资源;第二步,读入进程资源,建立进程链表,输入-1结束初始化;第三步, 打印当前系统资源分配表。4-1 4-24.5.2安全性算法Safety_Algorithm的程序流程图,如上图4-2;第一步,初识化work,finish;第二步,判断异常情况;第三步,Reasonable(head,finish,work,m,n)函数,找到当前可执行的进程第四步,执行安全检查,并显示检查后结果,返回给flag。 4.5.3接受进程请求,试分配Attempt_Allocation的程序流程图,如下图4-3 第一步,p指针指向弧的结点; 第二步,p指向邻接表的首节点; 第三步,p指针下移,直到为空,4.5.4对试分配后的系统,进行安全性检查Safety_Algorithm的程序流程图,和4.5.2大致一样,唯一一点在于当找不到安全序列时,将本次试分配作废,恢复该次试分配之前的数据结构。如下图4-4 4-4回收资源 4-3 5 程序分析测试5.1分模块分析与测试 函数的书写分模块进行,每完成一个模块进行调试、测试直到该函数运行无误。 5.1.1初始化系统资源模块Init_process(process *head,int m,int* count)的测试 按提示输入,以-1结束整个初始化过程,并打印结果。上图的效果起初不理想,经过一些调整后,显示才比较理想。5.1.2 试分配模块Attempt_Allocation的测试 试分配模块,主要是在系统进过第一次安全检查后,对系统资源的一次尝试性分配,试分配完成后,相关的数据结构被修改。5.1.3安全模块Safety_Algorithm的调试(1)试分配前的安全算法,结果如果输出一个安全性序列,并且经过人工检查该安全性序列,确实有效,则该模块正确;如果系统不安全,打印出相关信息,返回上一层。效果如下图示: (2)试分配后的安全算法,结果如果输出一个安全性序列,并且经过人工检查该安全性序列,确实有效,则该模块正确;否则,之前的试分配作废,恢复试分配前的资源状态。5.2集成测试 各模块测试通过后,集成在一起测试,系统初始资源和模块测试时保持一致,以下是一组测设用例以及结果,基本包括了该算法的所有情况。鉴于整个过程的截图较大,所以省略,部分过程可参看上面模块测试截图。(1)p6:结果;无此进程(2)P1: Request(2,0,2)结果;系统不能满足(3)P1: Request(1,0,2)结果;两次安全性检查都通过,并打印出最终结果(4)P0: Request(0,2,0)结果;试分配后,系统不安全,试分配作废(5)P0: Request(0,1,0)结果; 两次安全性检查都通过,并打印出最终结果6 小结(1) 在银行家算法的课程设计中,首先对该算法的思想进行分析,大致需要几个模块。(2) 首先,在网上收集了一些关于银行家算法的资料,包括它的起源,以及在实际中多个领域的应用,加深了对它的理解。之后,确定自己设计的算法分四大模块。首先需要初识化系统资源,其次,安全性检查,再者,试分配,最后是试分配后的安全性检查。(3) 在程序测试中出现了很多问题,譬如,死循环,逻辑关系的设计不当,还有显示效果不理想等等。但通过查阅书本,对算法细节重新建立正确的认识后,在通过单步调试后,最终解决,界面也几经尝试,才达到理想的界面风格。(4) 在集成测试中,由于之前的模块测试做的比较扎实,所以相对只是一些细节上的问题,很快也达到了预期的效果。 参考文献1 严蔚敏 吴伟民,.数据结构(C语言版),1999,清华大学出版社; 2 汤小丹 梁红兵 哲凤屏 汤子嬴, 计算机操作系统(第三版)西安电子科技大学出版社;附录(源代码)/银行家算法#include<stdio.h>#include<stdlib.h>#include<string.h>#define M 3 / 系统资源的种类#define N 10 / 进程上限#define D12 %5d%5d%5d%5d%5d%5d%5d%5d%5d%5d%5d%5d /输入输出格式定义typedef struct my_process int num; /进程标号 int MaxM;/ 表示某个进程对某类资源的最大需求 int AllocationM;/ 表示某个进程已分配到某类资源的个数 int NeedM;/ 表示某个进程尚需要某类资源的个数 struct my_process* next;process;void Insret_Tail(process *head,process node) /尾插法建立进程链表process* p=(process*)malloc(sizeof(process);process* last=NULL;memcpy(p,&node,sizeof(process); /动态创建进程结点p->next=NULL;if(NULL=*head)*head=p;else /找表尾last=*head;while(last->next!=NULL)last=last->next;last->next=p;/插入链尾void Init_process(process *head,int m,int* count)/初始化系统资源int i,j=0;process node;printf("请初始化一组进程,进程编号从0开始,输入-1 结束输入:n");donode.num=j+;printf("请输入第 %d个进程信息 :n",node.num);printf("最大需求矩阵: ");scanf("%d",&node.Max0);if(node.Max0!=-1) /输入-1 结束输入for(i=1;i<m;i+)scanf("%d",&node.Maxi);printf("分配矩阵: ");for(i=0;i<m;i+)scanf("%d",&node.Allocationi);printf("需求矩阵: ");for(i=0;i<m;i+)scanf("%d",&node.Needi);Insret_Tail(head,node);/插入链尾(*count)+;/进程数加1elsebreak;while(1);void Print(process *head,int *avail,int m)/打印初识系统资源process* p=NULL;int i,j;char ch;p=head;if(NULL=p)printf("当前无进程 !n"); exit(0);elseprintf("| Process | Max | |Allocation| | Need | |Available|nn");printf("t");for(i=0;i<4;i+)ch='A'for(j=0;j<m;j+)printf("%4c",ch+);printf("t");puts("");while(p!=NULL)printf("%8.2d",p->num);for(j=0;j<m;j+)printf("%4d",p->Maxj);printf("t");for(j=0;j<m;j+)printf("%4d",p->Allocationj);printf("t");for(j=0;j<m;j+)printf("%4d",p->Needj);printf("t");for(j=0;j<m;j+)printf("%4d",availj);printf("n");p=p->next;puts("");process* Location(process* head,int pro_num)/进程定位函数,找到当前请求进程在进程链表中的位置,以便后面对其操作process *p=NULL;p=head;if(NULL=p)printf("error !n");/异常,当前链表为空return p;elsewhile(p!=NULL)if(p->num=pro_num)break;elsep=p->next;if(NULL=p)/无此进程,输入错误printf("无此进程 !n");return p;else return p;process* Attempt_Allocation(process* head,int *request,int *avail,int m)/试分配int num,i;process* p=NULL;printf("请输入进程编号: n");scanf("%d",&num);p=Location(head,num);if(p=NULL)printf("无此进程!n");return p;printf("请输入该进程的请求向量: n");for(i=0;i<m;i+)scanf("%d",&requesti);for(i=0;i<m;i+)/请求的合法性检验if(requesti<=p->Needi);elseprintf("该请求系统不能满足 !n");return NULL;for(i=0;i<m;i+)if(requesti<=availi);elseprintf("该请求系统不能满足 !n");return NULL;for(i=0;i<m;i+) /试分配,修改相关数据结构availi=availi - requesti;p->Allocationi=p->Allocationi + requesti;p->Needi=p->Needi - requesti;return p;process* Reasonable(process* head,int*finish,int*work,int m,int n)/找当前可执行的进程int i=0,j=0,count=0;process* p=NULL;while(1)if(finishi!=-1) /表示该进程未执行安全性检查p=Location(head,finishi);/定位该进程if(p!=NULL)for(j=0;j<m;j+)if(p->Needj>workj)break;elsecontinue;if(j=m)return p;elsei+; /当前进程检查没有通过,则进行下一个进程的检查elsei+; /当前进程已经检查过,则进行下一个进程的检查if(i=n)return NULL; /遍历所有进程都未找到,则跳出返回NULLvoid Alter_Data(process* p,int* work,int* finish,int recordM,int m)