内存空间的回收和分配

来源:互联网 发布:项目经理 知乎 编辑:程序博客网 时间:2024/05/16 13:56
Code:
  1. #include<istream>   
  2. #include<string.h>   
  3. #include<iomanip.h>   
  4.   
  5. using namespace std;   
  6.   
  7. const int maxJob=10;//定义空闲分区和已分配分区数组个数的最大值都为10   
  8.   
  9. typedef struct node   
  10. {   
  11.     int start;//起始地址   
  12.     int length;//一个空闲区或已分配区的长度   
  13.     char tag[20];   
  14. }job; //类型定义,定义job代替struct node类型   
  15.   
  16. job freetable[maxJob];//定义空闲区表   
  17. int free_quantity;    //空闲区个数   
  18.   
  19. job occupys[maxJob]; //定义已分配区表   
  20. int occupy_quantity; //已分配区个数   
  21.   
  22. void initial(int a) //定义初始化函数   
  23. {   
  24.     occupys[0].start=20;   
  25.     occupys[0].length=32;   
  26.     strcpy(occupys[0].tag,"A");   
  27.     occupys[1].start=52;   
  28.     occupys[1].length=20;   
  29.     strcpy(occupys[1].tag,"B");   
  30.     occupys[2].start=98;   
  31.     occupys[2].length=40;   
  32.     strcpy(occupys[2].tag,"C");   
  33.     occupys[3].start=148;   
  34.     occupys[3].length=22;   
  35.     strcpy(occupys[3].tag,"D");   
  36.   
  37.     if(a==1)  //最先适应算法时的空闲表   
  38.     {   
  39.     freetable[0].start=72;   
  40.     freetable[0].length=26;   
  41.     strcpy(freetable[0].tag,"free");  //长度为 26   
  42.   
  43.     freetable[1].start=138;   
  44.     freetable[1].length=10;   
  45.     strcpy(freetable[1].tag,"free");  //长度为 10   
  46.   
  47.     freetable[2].start=170;   
  48.     freetable[2].length=86;   
  49.     strcpy(freetable[2].tag,"free");  //长度为 86   
  50.   
  51.     free_quantity=3;   
  52.     occupy_quantity=4;   
  53.     }   
  54.   
  55.     if(a==2)//最优适应算法时的空闲表   
  56.     {   
  57.     freetable[0].start=138;   
  58.     freetable[0].length=10;   
  59.     strcpy(freetable[0].tag,"free");   
  60.     freetable[1].start=72;   
  61.     freetable[1].length=20;   
  62.     strcpy(freetable[1].tag,"free");   
  63.     freetable[2].start=170;   
  64.     freetable[2].length=86;   
  65.     strcpy(freetable[2].tag,"free");   
  66.     free_quantity=3;   
  67.     occupy_quantity=4;   
  68.     }   
  69.   
  70.     if(a==3)//最差适应算法时的空闲表   
  71.     {   
  72.     freetable[0].start=170;   
  73.     freetable[0].length=86;   
  74.     strcpy(freetable[0].tag,"free");   
  75.     freetable[1].start=72;   
  76.     freetable[1].length=26;   
  77.     strcpy(freetable[1].tag,"free");   
  78.     freetable[2].start=138;   
  79.     freetable[2].length=10;   
  80.     strcpy(freetable[2].tag,"free");   
  81.     free_quantity=3;   
  82.     occupy_quantity=4;}   
  83.     }   
  84.   
  85. void view() //显示函数   
  86. {   
  87.     int i; //自增变量i   
  88.   
  89.     //输出当前空闲表   
  90.     cout<<endl<<"========================================================="<<endl;   
  91.     cout<<"当前空闲表:"<<endl;   
  92.     cout<<"起始地址   长度       状态"<<endl;   
  93.     for(i=0;i<free_quantity;i++)   
  94.     {   
  95.     //cout.setf(2);   
  96.     cout.width(12); //用width()函数更改当前的宽度设置。width()函数仅影响下一个显示项。之后,域宽度恢复成默认值。   
  97.     cout<<freetable[i].start;   
  98.     cout.width(10);   
  99.     cout<<freetable[i].length;   
  100.     cout.width(8);   
  101.     cout<<freetable[i].tag<<endl;   
  102.     }   
  103.     cout<<endl<<"当前总共有"<<free_quantity<<"个空闲区"<<endl;   
  104.   
  105.     //输出当前已分配表   
  106.     cout<<endl<<"----------------------------------------------------------"<<endl;   
  107.     cout<<"当前已分配表:"<<endl;   
  108.     cout<<"起始地址   长度     占用作业名"<<endl;   
  109.     for(i=0;i<occupy_quantity;i++)   
  110.     {   
  111.         //cout.setf(2);   
  112.         cout.width(12);   
  113.         cout<<occupys[i].start;   
  114.         cout.width(10);   
  115.         cout<<occupys[i].length;   
  116.         cout.width(8);   
  117.         cout<<occupys[i].tag<<endl;   
  118.     }   
  119.     cout<<endl<<"当前总共有"<<occupy_quantity<<"个已分配区"<<endl;   
  120. }//end view()   
  121.   
  122. //---------------------------------------最先适应分配算法---------------   
  123. void earliest()   
  124. {   
  125.    char job_name[20];//新建作业名   
  126.    int job_length;//新建作业名的长度   
  127.   
  128.    cout<<"请输入新申请内存空间的作业名和空间大小(最先适应):";   
  129.    cin>>job_name;//输入新作业的名字   
  130.    cin>>job_length;//输入新作业的长度   
  131.   
  132.    int i,flag=0; //flag是作为作业有没装入内存的标志,i作为数组元素的下标   
  133.   
  134.    for(i=0;i<free_quantity;i++)//顺序查找现存的空闲区,有没大于新作业大小的,如果有,就置flag为1 :能够装的下   
  135.    {   
  136.      if(freetable[i].length>=job_length)   
  137.      flag=1; //如果有能放得下的空闲区,把flag置1,说明此作业已经被放入内存中   
  138.    }   
  139.   
  140.   
  141.    if(flag==0)   
  142.    {   
  143.         cout<<endl<<"Sorry,当前没有能满足你申请长度的空闲内存,请稍候再试"<<endl;   
  144.    }   
  145.    else//下面情况是当发现适合的空闲区之后系统要做的操作   
  146.    {   
  147.          int t=0;   
  148.          i=0;   
  149.          while(t==0)//此while循环是判断当发现有装得下新作业的空闲区之后就不再搜索后面的空闲区,有发现适合的空闲区,t置1   
  150.         {   
  151.             if(freetable[i].length>=job_length)   
  152.                 {   
  153.                           t=1;   
  154.                 }   
  155.             i++;   
  156.         }   
  157.          //下面进行初始化新装入的作业   
  158.          i--; //为所发现的适合空闲区的数组元素(freetable[i])做准备,因为数组下标是从0开始,所以要减1   
  159.          occupys[occupy_quantity].start=freetable[i].start;//把发现的空闲区的起始地址变为新的已分配数组元素的起始地址   
  160.          strcpy(occupys[occupy_quantity].tag,job_name);//把新的已分配数组元素的名字变为新作业的名字   
  161.          occupys[occupy_quantity].length=job_length;//把新的已分配数组元素的长度变为新作业的长度   
  162.          occupy_quantity++;//把已分配区的个数加一   
  163.   
  164.   
  165.   
  166.          if(freetable[i].length>job_length)//如果所找到的能存放新作业的空闲区的长度大于新作业的长度   
  167.         {   
  168.             freetable[i].start+=job_length;//把这个空闲区的起始地址变后移新作业长度的距离   
  169.             freetable[i].length-=job_length;//把这个空闲区的长度减去新作业的长度   
  170.         }   
  171.          else//下面的情况是所找到的空闲区的长度刚好等于新作业的长度的情况   
  172.         {  int j;   
  173.             for(j=i;j<free_quantity-1;j++)   
  174.             {   
  175.                freetable[j]=freetable[j+1];   
  176.             }   
  177.             free_quantity--;   
  178.             cout<<endl<<"空闲区刚好被完全利用!"<<endl;   
  179.         }   
  180.    }   
  181. }   
  182.   
  183.   
  184. //--------------------------------------最优适应分配算法------------------   
  185. void excellent()   
  186. {   
  187.     char job_name[20];//新建作业名   
  188.     int  job_length; //新建作业名的长度   
  189.   
  190.     cout<<"请输入新申请内存空间的作业名和空间大小(最优适应):";   
  191.     cin>>job_name;//输入新作业的名字   
  192.     cin>>job_length;//输入新作业的长度   
  193.   
  194.     int i,flag=0;//flag是作为作业有没装入内存的标志,i作为数组元素的下标   
  195.     for(i=0;i<free_quantity;i++)//顺序查找现存的空闲区,有没大于新作业大小的,如果有,就置flag为1   
  196.     {   
  197.        if(freetable[i].length>=job_length)   
  198.        flag=1; //如果有能放得下的空闲区,把flag置1,说明此作业已经被放入内存中   
  199.     }   
  200.   
  201.     if(flag==0)   
  202.     {   
  203.        cout<<endl<<"Sorry,当前没有能满足你申请长度的空闲内存,请稍候再试"<<endl;   
  204.     }   
  205.     else  
  206.   {   
  207.        int t=0;   
  208.        i=0;   
  209.        while(t==0)   
  210.         {   
  211.           if(freetable[i].length>=job_length)//为下面的i做准备   
  212.            {   
  213.             t=1;   
  214.            }   
  215.           i++;   
  216.         }   
  217.        i--;//为所发现的适合空闲区的数组元素(freetable[i])做准备,因为数组下标是从0开始,所以要减1   
  218.   
  219.        //下面是最优算法的关键代码   
  220.       int j;   
  221.        for(j=0;j<free_quantity;j++)   
  222.        {   
  223.           if((freetable[j].length>=job_length)&&(freetable[j].length<freetable[i].length))   
  224.             {   
  225.                  i=j;   
  226.             }   
  227.        } //此循环结束之后j已经遍历所有空闲区找到一个最小的且能适合放新作业的空闲区freetable[i]   
  228.   
  229.        occupys[occupy_quantity].start=freetable[i].start;//把发现的空闲区的起始地址变为新的已分配数组元素的起始地址   
  230.        strcpy(occupys[occupy_quantity].tag,job_name);//把新的已分配数组元素的名字变为新作业的名字   
  231.   
  232.        occupys[occupy_quantity].length=job_length;//把新的已分配数组元素的长度变为新作业的长度   
  233.        occupy_quantity++;//把已分配区的个数加一   
  234.   
  235.   
  236.        if(freetable[i].length>job_length)   
  237.    {   
  238.           freetable[i].start+=job_length;   
  239.           freetable[i].length-=job_length;   
  240.    }   
  241.        else  
  242.    {   
  243.           for(j=i;j<free_quantity-1;j++)   
  244.   {   
  245.              freetable[j]=freetable[j+1];   
  246.   }   
  247.           free_quantity--;   
  248.           cout<<endl<<"空闲区刚好被完全利用!"<<endl;   
  249.    }   
  250. }   
  251. //end excellent()   
  252.   
  253. //-------------------------------------最坏适应算法-----------------------   
  254. void worst()   
  255. {   
  256.    char job_name[20]; //新建作业名   
  257.    int job_length;//新建作业名的长度   
  258.   
  259.    cout<<"请输入新申请内存空间的作业名和空间大小(最坏适应):";   
  260.    cin>>job_name;   
  261.    cin>>job_length;   
  262.   
  263.    int flag=0, i;   
  264.    for(i=0;i<free_quantity;i++)   
  265.    {   
  266.          if(freetable[i].length>=job_length)   
  267.            {   
  268.              flag=1;   
  269.            }   
  270.    }   
  271.   
  272.   
  273.    if(flag==0)   
  274.    {   
  275.        cout<<endl<<"Sorry,当前没有能满足你申请长度的空闲内存,请稍候再试"<<endl;   
  276.    }   
  277.    else  
  278.    {   
  279.        int t=0,i=0;   
  280.        while(t==0)   
  281.    {   
  282.          if(freetable[i].length>=job_length)   
  283.            {   
  284.              t=1;   
  285.            }   
  286.          i++;   
  287.    }   
  288.        i--;   
  289.    //下面代码是最坏适应算法的关键   
  290.    int j;   
  291.        for(j=0;j<free_quantity;j++)   
  292.    {   
  293.            if((freetable[j].length>=job_length)&&(freetable[j].length>freetable[i].length))   
  294.    {   
  295.                i=j;   
  296.    }   
  297.    }   
  298.   
  299.        occupys[occupy_quantity].start=freetable[i].start;   
  300.        strcpy(occupys[occupy_quantity].tag,job_name);   
  301.        occupys[occupy_quantity].length=job_length;   
  302.        occupy_quantity++;   
  303.   
  304.        if(freetable[i].length>job_length)   
  305.    {   
  306.             freetable[i].start+=job_length;   
  307.             freetable[i].length-=job_length;   
  308.    }   
  309.        else  
  310.    {   
  311.             for(j=i;j<free_quantity-1;j++)   
  312.                {   
  313.                    freetable[j]=freetable[j+1];   
  314.                }   
  315.             free_quantity--;   
  316.             cout<<endl<<"空闲区刚好被完全利用!"<<endl;   
  317.    }   
  318.    }   
  319. //end worst()   
  320.   
  321. //-------------------------------------撤消作业函数------------------------   
  322. void undoBlock()   
  323. {   
  324.     char job_name[20];//要删除的作业名   
  325.     int i,j;   
  326.   
  327.     cout<<"请输入要撤消的作业名:";   
  328.     cin>>job_name;   
  329.   
  330.     int flag=-1;   
  331.     int start,length;   
  332.   
  333.         for(i=0;i<occupy_quantity;i++)//按名顺序查找已有的已分配区,找出要删除的作业   
  334.     {   
  335.           if(!strcmp(occupys[i].tag,job_name))   
  336.           //如果if条件满足的话,   
  337.           //说明strcmp(occupys[i].tag,job_name)返回值为0,说明找到了,   
  338.           //条件是如够找到的话,就执行下面的语句   
  339.       {   
  340.             flag=i;   
  341.             start=occupys[i].start;//用start存放找到的要撤销的作业的起始地址   
  342.             length=occupys[i].length;//用length存放找到的要撤销的作业的长度   
  343.       }   
  344.     }   
  345.   
  346.     int p=0;//如果要删除的作业无相邻的空闲区,则把p置0;如果有,则把p置1   
  347.     if(flag==-1)   
  348.     {   
  349.             cout<<"没有这个作业名"<<endl;   
  350.     }   
  351.     else//下面是找到的情况   
  352.      {   
  353.   
  354.         //加入空闲表   
  355.         for(i=0;i<free_quantity;i++) //顺序查找空闲区表,并修改空闲区表   
  356.           {   
  357.             if((freetable[i].start+freetable[i].length)==start)//如果此要撤销的作业有上邻空闲区   
  358.                {   
  359.                   if(((i+1)<free_quantity)&&(freetable[i+1].start==start+length))//如果此要撤销的作业也有下邻空闲区   
  360.   
  361.                       {   
  362.                            freetable[i].length=freetable[i].length+freetable[i+1].length+length;   
  363.   
  364.                            for(j=i+1;j<free_quantity;j++)   
  365.                            {   
  366.                                 freetable[j]=freetable[j+1];//后面的空闲区表前移   
  367.                            }   
  368.   
  369.                            free_quantity--;   
  370.                            p=1;   
  371.                       }   
  372.                   else//如果此要撤销的作业只有上邻空闲区   
  373.                      {   
  374.                        freetable[i].length+=length;   
  375.                        p=1;   
  376.                      }   
  377.                 }   
  378.   
  379.             if(freetable[i].start==(start+length))//如果此要撤销的作业只有下邻空闲区   
  380.                   {   
  381.                      freetable[i].start=start;//相当于 删掉此已分配空间块   
  382.                      freetable[i].length+=length;   
  383.                      p=1;   
  384.                   }   
  385.           }   
  386.   
  387.   
  388.             if(p==0)//如果要删除的作业无相邻的空闲区   
  389.         {   
  390.                 freetable[free_quantity].start=start;   
  391.                 freetable[free_quantity].length=length;   
  392.                 free_quantity++;   
  393.         }   
  394.   
  395.   
  396.   
  397.         //删除已分配表中的该作业   
  398.             for(i=flag;i<occupy_quantity;i++)   
  399.         {   
  400.                   occupys[i]=occupys[i+1];   
  401.         }   
  402.             occupy_quantity--;   
  403.       }//end else   
  404. }//end finished()   
  405.   
  406.   
  407. int main()   
  408. {   
  409.     int t=1;   
  410.     int chioce=0;   
  411.     int flag=1;   
  412.     int a;   
  413.     state:   
  414.     cout<<endl<<endl<<"========================================================="<<endl;   
  415.     cout<<endl<<endl;   
  416.     cout<<"┏━━━━━━━━━━━━━━━━━━━━━━━┓"<<endl;   
  417.     cout<<"┃              算法模拟                 ┃"<<endl;   
  418.     cout<<"┣━━━━━━━━━━━━━━━━━━━━━━━┫"<<endl;   
  419.     cout<<"┃                实现                    ┃"<<endl;   
  420.     cout<<"|_______________________________________________|"<<endl;   
  421.     cout<<endl<<endl;   
  422.     cout<<"========================================================="<<endl;   
  423.     cout<<"请选择要采用的算法"<<endl<<endl;   
  424.     cout<<"1.最先适应算法 "<<endl<<"2.最优适应算法"<<endl<<"3.最差适应算法"<<endl;   
  425.     cout<<"----------------------------------------------------------"<<endl;   
  426.     cout<<"请选择:";   
  427.     cin>>a;   
  428.     while(a<1||a>3)   
  429.     {   
  430.     cout<<"选择错误!请选择1-3之间的整数!"<<endl<<"请重新输入:";   
  431.         cin>>a;   
  432.     }   
  433.     initial(a);   
  434.     while(flag==1)   
  435.     {   
  436.          cout<<endl<<endl<<"========================================================="<<endl<<endl;   
  437.          cout<<"请选择要进行的操作:"<<endl<<endl<<"1.显示空闲表和分配表" <<endl   
  438.          <<"2.撤消作业" <<endl<<"3.申请空间"<<endl<<"0.退出"<<endl<<endl;   
  439.          cout<<"----------------------------------------------------------"<<endl;   
  440.          cout<<"请选择:";   
  441.          cin>>chioce;   
  442.   
  443.         switch(chioce)   
  444.         {   
  445.            case 3:   
  446.            switch(a)   
  447.            {   
  448.               case 1: earliest(); break;   
  449.               case 2: excellent();break;   
  450.               case 3: worst();break;   
  451.            } // end switch(a)   
  452.            break;   
  453.   
  454.            case 2:   
  455.            undoBlock();   
  456.            break;   
  457.   
  458.            case 1:   
  459.            view();   
  460.            break;   
  461.   
  462.            case 0:   
  463.            goto state;   
  464.            break;   
  465.   
  466.            default:   
  467.            cout<<"选择错误!请重新输入"<<endl;   
  468.         }//end switch(chioce)   
  469.     }//end while()   
  470. }//end mian()   

 

原创粉丝点击