银行家算法(操作系统)

一、实验目的

通过编写银行家算法,要求学生进一步掌握如何实现死锁的避免,进一步熟练使用数组进行程序的设计及实现。

要求:对教材128页的31题,请用银行家算法完成(1)和(2)

二、实验原理及基本技术路线图(方框原理图)

用C语言或C++语言开发。实现银行家算法、安全性检测算法。

请加上算法的详细描述。程序功能结构图、流程图、数据结构定义、主要变量的说明、函数的说明等

流程图:

三、所用仪器、材料(设备名称、型号、规格等)。

计算机一台

四、实验方法、步骤

代码:

#include<stdio.h>
#include<stdlib.h>
 
#define False 0
#define True 1
 
/********主要数据结构********/
char NAME[100]={0};//资源的名称
int Max[100][100]={0};//最大需求矩阵
int Allocation[100][100]={0};//系统已分配矩阵
int Need[100][100]={0};//还需要资源矩阵
int Available[100]={0};//可用资源矩阵
int Request[100]={0};//请求资源向量
int Work[100]={0};//存放系统可提供资源量
int Finish[100]={0}; //标记系统是否有足够的资源分配给各个进程
int Security[100]={0};//存放安全序列
 
int M=100;//进程的最大数
int N=100;//资源的最大数
 
/********初始化数据:输入进程数量、资源种类、各种资源可利用数量、
各进程对资源最大需求量、各进程的资源已分配数量等。********/
void init()
{
    /* m为进程个数,即矩阵行数,n为资源种类,即矩阵列数。*/
    int i,j,m,n;
    int number,flag;
    char name;//输入资源名称
    int temp[100]={0};//统计已经分配的资源
    //输入系统资源数目及各资源初试个数
    printf("系统可用资源种类为:");
    scanf("%d",&n);
    N=n;
    for(i=0;i<n;i++)
    {
        printf("资源%d的名称:",i);
        fflush(stdin);  //清空输入流缓冲区的字符,注意必须引入#include<stdlib.h>头文件
        scanf("%c",&name);
        NAME[i]=name;
        printf("资源%c的初始个数为:",name);
        scanf("%d",&number);
        Available[i]=number;
    }
 
    //输入进程数及各进程的最大需求矩阵
    printf("\n请输入进程的数量:");
    scanf("%d",&m);
    M=m;
    printf("请输入各进程的最大需求矩阵的值[Max]:\n");
    do{
        flag = False;
        for(i=0;i<M;i++)
            for(j=0;j<N;j++)
            {
                scanf("%d",&Max[i][j]);
                if(Max[i][j]>Available[j])
                    flag = True;
            }
        if(flag)
            printf("资源最大需求量大于系统中资源最大量,请重新输入!\n");
    } while(flag);
 
 
    //输入各进程已经分配的资源量,并求得还需要的资源量
    do{
        flag=False;
        printf("请输入各进程已经分配的资源量[Allocation]:\n");
        for(i=0;i<M;i++)
        {
            for(j=0;j<N;j++)
              {
                scanf("%d",&Allocation[i][j]);
                if(Allocation[i][j]>Max[i][j])
                    flag=True;
                Need[i][j]=Max[i][j]-Allocation[i][j];
                temp[j]+=Allocation[i][j];//统计已经分配给进程的资源数目
              }
        }
        if(flag)
            printf("分配的资源大于最大量,请重新输入!\n");
    }while(flag);
 
    //求得系统中可利用的资源量
    for(j=0;j<N;j++)
        Available[j]=Available[j]-temp[j];
}
 
/********显示资源分配矩阵********/
void showdata()
{
    int i,j;
    printf("*************************************************************\n");
    printf("系统目前可用的资源[Available]:\n");
    for(i=0;i<N;i++)
        printf("%c  ",NAME[i]);
    printf("\n");
    for(j=0;j<N;j++)
        printf("%d  ",Available[j]);
    printf("\n");
    printf("系统当前的资源分配情况如下:\n");
    printf("            Max        Allocation    Need\n");
    printf("进程名      ");
    //Max、Allocation、Need下分别对应
    for(j=0;j<3;j++){
        for(i=0;i<N;i++)
            printf("%c  ",NAME[i]);
        printf("     ");
    }
    printf("\n");
    //输出每个进程的Max、Allocation、Need
    for(i=0;i<M;i++){
        printf(" P%d        ",i);
        for(j=0;j<N;j++)
            printf("%d  ",Max[i][j]);
        printf("     ");
        for(j=0;j<N;j++)
            printf("%d  ",Allocation[i][j]);
        printf("     ");
        for(j=0;j<N;j++)
            printf("%d  ",Need[i][j]);
        printf("\n");
    }
}
 
/********尝试分配资源********/
int test(int i) //试探性的将资源分配给第i个进程
{
    for(int j=0;j<N;j++)
    {
        Available[j]=Available[j]-Request[j];
        Allocation[i][j]=Allocation[i][j]+Request[j];
        Need[i][j]=Need[i][j]-Request[j];
    }
    return True;
}
 
/********试探性分配资源作废********/
int Retest(int i) //与test操作相反
{
    for(int j=0; j<N; j++)
    {
        Available[j] = Available[j] + Request[j];
        Allocation[i][j] = Allocation[i][j] - Request[j];
        Need[i][j] = Need[i][j] + Request[j];
    }
    return True;
}
 
/********安全性算法********/
int safe()
{
    int i,j,k=0,m,apply;
    //初始化work
    for(j=0;j<N;j++)
        Work[j] = Available[j];
    //初始化Finish
    for(i=0;i<M;i++)
        Finish[i] = False;
    //求安全序列
    for(i=0;i<M;i++){
        apply=0;
        for(j=0;j<N;j++){
            if(Finish[i]==False && Need[i][j]<=Work[j])
            {
                apply++;
                //直到每类资源尚需数都小于系统可利用资源数才可分配
                if(apply==N)
                {
                    for(m=0;m<N;m++)
                        Work[m]=Work[m]+Allocation[i][m];//更改当前可分配资源
                    Finish[i]=True;
                    Security[k++]=i;
                    i=-1; //保证每次查询均从第一个进程开始
                }
            }
        }
    }
 
    for(i=0;i<M;i++){
        if(Finish[i]==False){
            printf("系统不安全\n");//不成功系统不安全
            return False;
        }
    }
    printf("系统是安全的!\n");//如果安全,输出成功
    printf("存在一个安全序列:");
    for(i=0;i<M;i++){//输出运行进程数组
        printf("P%d",Security[i]);
        if(i<M-1)
            printf("->");
    }
    printf("\n");
    return True;
}
 
/********利用银行家算法对申请资源进行试分********/
void bank()
{
    int flag = True;//标志变量,判断能否进入银行家算法的下一步
    int i,j;
 
    printf("请输入请求分配资源的进程号(0-%d):",M-1);
    scanf("%d",&i);//输入须申请资源的进程号
 
    printf("请输入进程P%d要申请的资源个数:\n",i);
    for(j=0;j<N;j++)
    {
        printf("%c:",NAME[j]);
        scanf("%d",&Request[j]);//输入需要申请的资源
    }
 
    //判断银行家算法的前两条件是否成立
    for (j=0;j<N;j++)
    {
        if(Request[j]>Need[i][j])//判断申请是否大于需求,若大于则出错
        {
            printf("进程P%d申请的资源大于它需要的资源",i);
            printf("分配不合理,不予分配!\n");
            flag = False;
            break;
        }
        else
        {
            if(Request[j]>Available[j])//判断申请是否大于当前可分配资源,若大于则出错
            {
                printf("进程%d申请的资源大于系统现在可利用的资源",i);
                printf("\n");
                printf("系统尚无足够资源,不予分配!\n");
                flag = False;
                break;
            }
        }
    }
    //前两个条件成立,试分配资源,寻找安全序列
    if(flag) {
        test(i); //根据进程需求量,试分配资源
        showdata(); //根据进程需求量,显示试分配后的资源量
        if(!safe()) //寻找安全序列
        {
            Retest(i);
            showdata();
        }
    }
}
 
 
int main()//主函数
{
    char choice;
    init();//初始化数据
    showdata();//显示各种资源
    //用银行家算法判定系统当前时刻是否安全,不安全就不再继续分配资源
    if(!safe()) exit(0);
 
    do{
        printf("*************************************************************\n");
        printf("\n");
        printf("\n");
        printf("\t-------------------银行家算法演示------------------\n");
        printf("                     R(r):请求分配   \n");
        printf("                     E(e):退出       \n");
        printf("\t---------------------------------------------------\n");
        printf("请选择:");
        fflush(stdin);  //清空输入流缓冲区的字符,注意必须引入#include<stdlib.h>头文件
        scanf("%c",&choice);
        switch(choice)
        {
            case 'r':
            case 'R':
                bank();break;
            case 'e':
            case 'E':
                exit(0);
            default: printf("请正确选择!\n");break;
        }
    } while(choice);
}

五、实验过程原始记录(数据、图表、计算等)

六、实验结果、分析和结论(误差分析与数据处理、成果总结等。其中,绘制曲线图时必须用计算纸)

对整个上机过程、上机结果进行总结、分析。包括收获、存在问题、改进等方面。

银行家算法中的数据结构

(1)可利用资源向量Available

这是个含有m个元素的数组,其中的每一个元素代表一类可利用的资源数目。其初始值是系统中所配置的该类全部可用资源的数目,其数值随该类资源的分配和回收而动态地改变,如果Available[j]=K,则表示系统中现有Rj 类资源的最大数目为K。

(2)最大需求矩阵Max

这是一个n×m的矩阵,它定义了系统中n个进程中的每一个进程对m类资源的最大需求。如果Max[i,j]=K,则表示进程i需要Rj 类资源的最大数目为K。

(3)分配矩阵Allocation

这也是一个n×m的矩阵,它定义了系统中每一类资源当前已分配给每一进程的资源数。如果Allocation[i,j]=K,则表示进程i当前已分得Rj 类资源的数目为K。

(4)需求矩阵Need。

这也是一个n×m的矩阵,用以表示每一个进程尚需的各类资源数。如果Need[i,j]=K,则表示进程i还需要Rj 类资源K个方能完成其任务。

上述三个矩阵间存在下述关系:

Need[i,j]=Max[i,j]-Allocation[i,j]

2. 银行家算法

设Requesti是进程Pi 的请求向量,如果Requesti[j]=K,表示进程Pi 需要K个Rj 类型的资源。当Pi 发出资源请求后,系统按下述步骤进行检查:

(1)如果Requesti[j]≤Need[i,j],便转向步骤(2);否则认为出错,因为它所需要的资源数已超过它所宣布最大值。

(2)如果Requesti[j]≤Available[j],便转向步骤(3);否则,表示尚无足够资源, Pi须等待。

(3)系统试探着把资源分配给进程Pi ,并修改下面数据结构中的数值

Available[j]=Available[j]-Requesti[j];

Allocation[i,j]=Allocation[i,j]+Requesti[j];

Need[i,j]=Need[i,j]-Requesti[j];

(4)系统执行安全性算法,检查此次资源分配后,系统是否处于安全状态。若安全,才正式将资源分配给进程Pi ,以完成本次分配;否则,将本次的试探分配作废,恢复原来的资源分配状态,让进程Pi 等待。

3. 安全性算法

(1)设置两个向量:

工作向量Work: 它表示系统可提供给进程继续运行所需的各类资源数目,它含有m个元素,在执行安全算法开始时,Work=Available;

Finish: 它表示系统是否有足够的资源分配给进程,使之运行完成。开始时先做Finish[i]=false; 当有足够资源分配给进程时, 再令Finish[i]=true。

(2)从进程集合中找到一个能满足下述条件的进程:

Finish[i]=false;

Need[i,j]≤Work[j];

若找到,执行(3),否则,执行 (4)

(3)当进程Pi 获得资源后,可顺利执行,直至完成,并释放出分配给它的资源,故应执行:

Work[j]= Work[i]+Allocation[i,j];

Finish[i]=true;

go to step 2;

(4)如果所有进程的Finish[i]=true都满足, 则表示系统处于安全状态;否则,系统处于不安全状态;

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

cramyyy

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值