基于网格的DAG任务调度算法实现(BNP-ISH、mcp、dls、etf)

        08年5月份 接的一个本科毕业设计,用了2天看懂了第一个算法(BNP-ISH),之后连续实现了其他的几个(mcp、dls、etf),第一个懂了以后后面的就好做很多,每个算法200多¥。。。。该生顺利毕业。呵呵。

        这次硬盘之灾,让我心有余悸,我想有些东西应该整理下,记录下来,对大家用处不大,对自己来说,好歹是个工作学习的见证。。。

 

 

  1. /*************************************************************************************************/
  2. //                                                                                      BNP-ISH算法:                                                                                                                                                                                      
  3. //      1)定义任务图,本程序采用《并行分布计算中的调度算法理论与设计》P62图4.1的任务图。
  4. //      2)计算每一个节点t到最终节点的最长路经的b_level值,确定优先级,b_level值越大越优先
  5. //      3)从没有前向节点的节点开始执行,执行就是将节点i分配到n台机器中最优(可最早执行的)一台上,
  6. //           每一台机器的执行任务速度不一样,可设置。
  7. //      4)执行完成之后,再来执行。
  8. //
  9. //ISH算法:[ISH算法的时间复杂度是O(N*N)]
  10. //      1)计算每个节点的SBL属性;
  11. //      2)按SBL降序构造一个就绪任务表。
  12. //      开始时,就绪任务表仅包含入口节点,然后重复如下步骤直到所有节点都调度完毕:
  13. //      (1)用插入法将就绪表中的第一个节点调度到能够最早执行的处理机上;
  14. //      (2)如果调度这个节点产生一个空闲段,则在就绪表中寻找尽可能多的节点,这些节点可以调度到这些空闲的时间片,
  15. //           但不能更早调度到其他处理机上
  16. //      (3)把已经就绪的节点插入到就绪表中
  17. //
  18. //
  19. //
  20. //
  21. //
  22. /*************************************************************************************************/
  23. #include "stdio.h"
  24. int max(int SBL[10])//该函数实现按SBL属性值的降序顺序从任务列表中取任务
  25. {
  26. int maxsbl=0;
  27. int maxid;
  28. int i;
  29. for(i=1;i<10;i++)
  30. {
  31. if(SBL[i]>maxsbl){maxsbl=SBL[i];maxid=i;}
  32. }
  33. return maxid;
  34. }
  35. int maxitem(int a[4])//返回长度为4的数组中最大的项的下标
  36. {
  37.     int k=a[0]>a[1]?0:1;
  38.     k= a[k]>a[2]?k:2;
  39.     return a[k]>a[3]?k:3;
  40. }
  41. int minitem(int a[4])//返回长度为4的数组中最小项的下标【如果有最小项有相等的情况,则小下标优先】
  42. {
  43.     int k=a[3]<a[2]?3:2;
  44.     k=a[k]<a[1]?k:1;
  45.     return a[k]<a[0]?k:0;
  46. }
  47. void main()
  48. {
  49.     //定义任务图,采用二维数组表示所有节点间的关系,例如t1节点指向t2则node[1][2] = 4,而node[1][9] = 0,因为t1没有指向t9的路经
  50.     //由于没有节点t0则T[0][i]全部为0。
  51.     int i;
  52.     int j;
  53.     int node[10][10] = {0};
  54.     node[1][2] = 4; 
  55.     node[1][3] = 1;
  56.     node[1][4] = 1;
  57.     node[1][5] = 1;
  58.     node[1][7] = 10;
  59.     node[2][6] = 1;
  60.     node[2][7] = 1;
  61.   
  62.     node[3][8] = 1;
  63.     node[4][8] = 1;
  64.     
  65.     node[6][9] = 5;
  66.     
  67.     node[7][9] = 6;
  68.     
  69.     node[8][9] = 5;
  70. /********************************************************************************/  
  71.     //定义每一个节点的计算时间,忽略t[0]
  72.     int t[10];
  73.     t[1] = 2;
  74.     t[2] = 3;
  75.     t[3] = 3;
  76.     t[4] = 4;
  77.     t[5] = 5;
  78.     t[6] = 4;
  79.     t[7] = 4;
  80.     t[8] = 4;
  81.     t[9] = 1;
  82. /********************************************************************************/  
  83.     //定义每一个节点的前导节点,例如节点8的前导节点为节点3、4,则diaodao[8][3] = 1,diandao[8][4] = 1
  84.     int qiandao[10][10] = {0};
  85.     qiandao[2][1] = 1;
  86.     qiandao[3][1] = 1;
  87.     qiandao[4][1] = 1;
  88.     qiandao[5][1] = 1;
  89.     qiandao[7][1] = 1;
  90.     qiandao[6][2] = 1;
  91.     qiandao[7][2] = 1;
  92.     qiandao[8][3] = 1;
  93.     qiandao[8][4] = 1;
  94.     qiandao[9][6] = 1;
  95.     qiandao[9][7] = 1;
  96.     qiandao[9][8] = 1;
  97.         
  98. /********************************************************************************/  
  99.     //计算b_level值,忽略b_level[0],由于事先已知头节点和尾节点(在本例中头节点为t1,尾节点为t5和t9)
  100.     //所以从尾节点倒序计算每一个节点的b_level值。
  101.     int b_level[10] = {0};  
  102.     //尾节点b_level值为0
  103.     b_level[5] = 5;
  104.     b_level[9] = 1; 
  105.     //指示是否已经计算过b_level值
  106.     int b_cal[10] = {0};
  107.     b_cal[5] = 1;
  108.     b_cal[9] = 1;
  109.     int b_level_max, temp1, temp2, temp;
  110.     //从节点t1开始计算b_level值
  111.     
  112.     //从尾节点9开始倒序计算b_level值,直到头节点1结束,由于最多有三次向前计算的机会,所以算三次就结束。
  113.     for(int i = 1; i < 10; i++) 
  114.         if(qiandao[9][i] == 1)  
  115.             b_level[i] = t[9] + t[i] + node[i][9];  
  116.     //计算倒数第二排节点
  117.     for(int i = 1; i < 9; i++)
  118.     {
  119.         if(b_level[i] != 0)
  120.         {
  121.             for(int j = 1; j < 10; j++)
  122.                 if(qiandao[i][j] == 1)
  123.                 {
  124.                     temp1 = t[j] + node[j][i] + b_level[i];
  125.                     if(temp1 > b_level[j])
  126.                         b_level[j] = temp1;
  127.                 }
  128.         }
  129.     }   
  130.     //最后计算t1节点 
  131.     for(int i = 1; i < 10; i++)
  132.     {
  133.         if(qiandao[i][1] == 1)
  134.         {
  135.             temp1 = t[1] + node[1][i] + b_level[i];
  136.             if(temp1 > b_level[1])
  137.                 b_level[1] = temp1;
  138.         }
  139.         
  140.     }
  141.     
  142. /********************************************************************************/  
  143. /********************************************************************************/  
  144.     //计算b_level值,忽略b_level[0],由于事先已知头节点和尾节点(在本例中头节点为t1,尾节点为t5和t9)
  145.     //所以从尾节点倒序计算每一个节点的b_level值。
  146.     int SBL[10] = {0};  
  147.     //尾节点b_level值为0
  148.     SBL[5] = 5;
  149.     SBL[9] = 1; 
  150.     //指示是否已经计算过b_level值
  151.     int SBL_cal[10] = {0};
  152.     SBL_cal[5] = 1;
  153.     SBL_cal[9] = 1;
  154.     int SBL_max, SBL_temp1, SBL_temp2, SBL_temp;
  155.     //从节点t1开始计算b_level值
  156.     
  157.     //从尾节点9开始倒序计算b_level值,直到头节点1结束,由于最多有三次向前计算的机会,所以算三次就结束。
  158.     for(int i = 1; i < 10; i++) 
  159.         if(qiandao[9][i] == 1)  
  160.             SBL[i] =t[9] + t[i] ;       
  161.     //计算倒数第二排节点
  162.     for(int i = 1; i < 9; i++)
  163.     {
  164.         if(SBL[i] != 0)
  165.         {
  166.             for(int j = 1; j < 10; j++)
  167.                 if(qiandao[i][j] == 1)
  168.                 {
  169.                     SBL_temp1 = t[j]  + SBL[i];
  170.                     if(SBL_temp1 > SBL[j])
  171.                         SBL[j] = SBL_temp1;
  172.                 }
  173.         }
  174.     }   
  175.     //最后计算t1节点 
  176.     for(int i = 1; i < 10; i++)
  177.     {
  178.         if(qiandao[i][1] == 1)
  179.         {
  180.             SBL_temp1 =t[1]  + SBL[i];
  181.             if(SBL_temp1 > SBL[1])
  182.                 SBL[1] = SBL_temp1;
  183.         }
  184.         
  185.     }
  186.     
  187. //计算每个节点的前向大数
  188.     int qian_max[10] = {0};
  189.     for(i = 1; i < 10; i++)
  190.     {
  191.             temp1 = 0;
  192.             for(int j = 0; j < 10; j++)
  193.             {
  194.                 if(node[j][i] > qian_max[i])    
  195.                     qian_max[i] = node[j][i];
  196.             }           
  197.             
  198.     }
  199. /
  200. //计算节点t1到其他所有节点的最大路径值
  201.     int t_level[10] = {0};
  202.     for(i = 1; i < 10; i++)
  203.     {
  204.         if(node[1][i] != 0)
  205.         {
  206.             t_level[i] = t[1] + node[1][i];             
  207.             if(t_level[i] < temp1)
  208.                 t_level[i] = temp1;
  209.             for(int j = 1; j < 10; j++)
  210.             {
  211.                 if(node[i][j] != 0) 
  212.                 {
  213.                     t_level[j] = t_level[i] + node[i][j]+t[i];              
  214.                     for(int k = 1; k < 10; k++)
  215.                     {
  216.                         if(node[j][k] != 0) 
  217.                             t_level[k] = t_level[j] + node[j][k];
  218.                         
  219.                     }
  220.                 }
  221.             }   
  222.         }
  223.     }
  224.     for(i = 1; i < 10; i++)
  225.         printf("t_level[%d] = %d/n", i, t_level[i]);
  226.         
  227.         
  228.         for(i = 1; i < 10; i++)
  229.         printf("sbl[%d] = %d/n", i, SBL[i]);
  230.     for(i = 1; i < 10; i++)
  231.         printf("qian_max[%d] = %d/n", i, qian_max[i]);
  232.     for(i = 1; i < 10; i++) 
  233.     {printf("b_level[%d]: %d/n ",i, b_level[i]);}
  234.     
  235.     
  236.     
  237. ISH
  238.     
  239.     
  240. int pe[10]={-1,-1,-1,-1,-1,-1,-1,-1,-1,-1};//记录任务的开始时间,如pe[i]=x表示i任务开始执行时间为x
  241. int time[4]={0,0,0,0};                     //记录机器当前时间,其值为当前分配的任务开始时间+任务的执行时间
  242. int tdispatched[10]={-1,-1,-1,-1,-1,-1,-1,-1,-1,-1};//记录任务的分配情况,如tdispatched[i]=x表示i任务被分配到x号机器上
  243. for(i=1;i<10;i++)  //从9个任务里面,按sbl的降序取值
  244. {
  245. int maxid=max(SBL);
  246. SBL[maxid]=0;
  247. //=========================================================================================分配maxid开始
  248. if(maxid==1){
  249.         
  250.         pe[maxid]=0;
  251.         tdispatched[maxid]=0;
  252.         time[0]+=t[maxid];//0号机器被用掉的时间为该任务的运行耗时
  253.     
  254.     
  255.     
  256.     }else{
  257.         
  258.         
  259.         
  260.           
  261.         int early[4]={-1,-1,-1,-1};//记录一个任务如果分配到0-3号机器能够最早执行的时间
  262.     
  263. int a[4]={-1,-1,-1,-1};//记录由各个父节点产生的开始时间限制(包括同机器无通信时间情况与非同机器有通信时间情况)    
  264.     for(int machid=0;machid<4;machid++)//往4台机器上分
  265.     {
  266.   int j=1;
  267.   a[0]=-1;
  268.   a[1]=-1;
  269.   a[2]=-1;
  270.   a[3]=-1;
  271.   
  272.   //记录由各个父节点产生的开始时间限制(包括同机器无通信时间情况与非同机器有通信时间情况)
  273. //---------------------------------------------------------------------------------------------------------------trymap----
  274. a[0]=time[machid];//记录当前机器的当前时间,如果任务被分配当前机器,任务必须在该时间以后执行
  275. for(int i=1;i<10;i++)//遍历 找到maxid节点的前驱
  276.    {
  277.     if(node[i][maxid]!=0)//找到一个前驱,node[i][maxid]的值为节点maxid与节点i的通信时间
  278.       {
  279.          
  280.           
  281.           if(tdispatched[i]==machid)//如果前驱节点i与该节点maxid都被分到了同一个机器
  282.            {
  283.                
  284.                
  285.                a[j]=time[machid];//maxid任务开始时间紧接着该机器的当前时间
  286.                
  287.                j++;
  288.             }else{//如果前驱节点与该节点没有分配到同一个机器
  289.                a[j]=pe[i]+t[i]+node[i][maxid];//则maxid任务开始时间为前驱任务节点完成时间+与之通信的时间
  290.               
  291.                j++;
  292.             }
  293.     
  294.       }
  295.    }//for完成后,j为 maxid 节点 前驱的个数,a[0....(j-1)]为对应前驱节点产生的开始时间,其中最大的为必须满足的
  296.   
  297. early[machid]=a[maxitem(a)];
  298. //----------------------------------------------------------------------------------------------------------------trymap---
  299.     }   //得到early[0...3],里面最小的为选中的分配方案(开始时间最早)
  300.         
  301.     int fangan=minitem(early);//记录应该分配到哪台机器(开始时间最早)
  302.     tdispatched[maxid]=fangan;//tdispatched[maxid]记录maxid任务被分配到哪台机器
  303.     pe[maxid]=early[fangan];  //pe[maxid]记录maxid任务的开始时间
  304.     time[fangan]=pe[maxid]+t[maxid];
  305.     
  306.     
  307.     
  308.     }
  309. //==========================================================================================分配maxid结束
  310. }
  311. printf("【pe[i]表示i任务节点开始的执行时间】/n");
  312. for(i = 1; i < 10; i++) 
  313.     {printf("pe[%d]: %d/n ",i, pe[i]);}
  314. printf("【i任务节点被分配到编号为tdispatched[i]的机器】/n");
  315. for(i = 1; i < 10; i++) 
  316.     {printf("tdispatched[%d]: %d/n ",i, tdispatched[i]);}
  317. printf("【总共耗费的时间】");
  318.     {printf("%d/n ",time[maxitem(time)]);}
  319.     
  320.     
  321.  /*
  322.    
  323.     int   Start[9][4];
  324.     int   Flag[9];
  325.     int   jiuxu[9];
  326.     int   temp_z[9] = {0};
  327.     int   max;
  328.     int   start=0;
  329.     int   r[4];
  330.     int   SETC[9][4];
  331.     int    min;
  332.     
  333.     for(int j=1;j<5;j++)
  334.     {r[j]=0;}
  335.     int k = 1;
  336.     while(k<10)
  337.     {
  338.     for(i=1;i<10;i++)
  339.     {
  340.          Flag[i]=1;
  341.          jiuxu[i]=0;
  342.     }
  343.     
  344.     for(i=1;i<10;i++)
  345.     {   if(Flag[i]==1)
  346.         for(int j = 1; j < 5; j++)      
  347.             temp_z[i] += qiandao[i][j]; 
  348.            
  349.             
  350.     
  351.     }
  352.     for(i=1;i<10;i++)
  353.     {
  354.         if(Flag[i]==1&&temp_z[i]==0)
  355.             jiuxu[i]=1;
  356.     }
  357.     for(i=1;i<10;i++)
  358.     {
  359.         if(jiuxu[i]==1)
  360.             
  361.                 max=b_level[i];
  362.                     start=i;
  363.                 break;      
  364.     }
  365.     for(i=1;i<10;i++)
  366.     {
  367.         if(jiuxu[i]==1&&b_level[i]>max)
  368.             max=b_level[i];
  369.             start=i;
  370.     }
  371.         
  372.      
  373.       int m=1;
  374.       min=r[1];
  375.     for(j=1;j<5;j++)
  376.     {      
  377.             
  378.             if(r[j]<min)
  379.             
  380.                 m=j;
  381.                 
  382.         }
  383.     
  384.     r[m]=r[m]+t[start];
  385.     if(t_level[start]>r[m])
  386.                 r[m]=t_level[start]+t[start];
  387.             
  388.                 Flag[i]=0;
  389.                 
  390.     k++;
  391.     }
  392.     int rmax=0;
  393.    for(j=1;j<5;j++)
  394.    {
  395.        if(r[j]>rmax)
  396.            rmax=r[j];
  397.            
  398.    }
  399.    for(j=1;j<5;j++)
  400.    {    printf("r[%d]: %d/n ",j,r[j]);}
  401.    
  402. printf("完成时间为: %d/n ",rmax);
  403. */
  404. }

 

  1. /*************************************************************************************************/
  2. //                                                                                      BNP-MCP算法:                                                                                                                                                                                      
  3. //      1)定义任务图,本程序采用《并行分布计算中的调度算法理论与设计》P62图4.1的任务图。
  4. //      2)计算每一个节点t到最终节点的最长路经的b_level值,确定优先级,b_level值越大越优先
  5. //      3)从没有前向节点的节点开始执行,执行就是将节点i分配到n台机器中最优(可最早执行的)一台上,
  6. //           每一台机器的执行任务速度不一样,可设置。
  7. //      4)执行完成之后,再来执行。
  8. //
  9. //MCP算法:[MCP算法的时间复杂度是O(N*N*logn)]
  10. //      1)计算每个节点的ALAP属性;
  11. //      2)对每个节点,创建它本身的以及它的所有子结点的ALAP时间按降序构成的表;
  12. //      3)将这些表按字典排序,并按这个顺序创建一个节点表,然后反复执行如下步骤,直到节点表为空时停止
  13. //      (1)用插入法将将节点表的第一个节点调度到能够使该节点最早开始执行的处理机
  14. //      (2)从这个节点表中去掉这个节点
  15. //           
  16. //      
  17. //
  18. //
  19. //
  20. //
  21. //
  22. /*************************************************************************************************/
  23. #include "stdio.h"
  24. int numofchild(int *p)
  25. {
  26.     int length=0;
  27. for(int i=0;i<5;i++)
  28. {
  29.     if(p[i]!=-1){length++;}else{break;}
  30. }
  31. return length;
  32. }
  33. void BubbleSort(int *p)
  34. {
  35. int k=numofchild(p)-1;
  36. int swap;
  37. for(int i=1;i<k;i++)
  38. for(int j=k;j>i;j--)
  39. {   
  40.     if(p[i]<p[j])
  41.     {
  42.         swap=p[i];
  43.         p[i]=p[j];
  44.         p[j]=swap;
  45.     
  46.     
  47.     
  48.     }
  49. }
  50. }
  51. void  DictionarySort(int (*b)[5],int seq[10])//字典排序
  52. {
  53. int k=0;
  54. int a[9][5];
  55. for(int i=0;i<9;i++)
  56. for(int j=0;j<5;j++)
  57. a[i][j]=(*b)[k++]; 
  58. int *src[9]={a[0],a[1],a[2],a[3],a[4],a[5],a[6],a[7],a[8]};
  59. //-----------------------------------------------------------------------------------------------------------------------------------
  60.  int *p;
  61.  int l,m;
  62.  int x,y,swap;
  63. int n=9;
  64.  for (int i=0;i<n-1;i++)
  65.  {
  66.   l=numofchild(src[i]);
  67.   for (int j=i+1;j<n;j++)
  68.   {
  69.      m=numofchild(src[j]); 
  70.      for (x=0,y=0;(x<l)&&(y<m);x++,y++)
  71.      {
  72.       
  73.        if ((src[i][x]>src[j][y]))
  74.        {
  75.         p=src[i];
  76.         src[i]=src[j];
  77.         src[j]=p;
  78.        swap=seq[i+1];
  79.        seq[i+1]=seq[j+1];
  80.        seq[j+1]=swap;
  81.         break;
  82.        }else if(src[i][x]<src[j][y])
  83.        {
  84.        break;
  85.        
  86.        }
  87.      
  88.      
  89.      }
  90.      if(y==m)
  91.      {
  92.      p=src[i];
  93.         src[i]=src[j];
  94.         src[j]=p;
  95.         swap=seq[i+1];
  96.        seq[i+1]=seq[j+1];
  97.        seq[j+1]=swap;
  98.        break;
  99.         
  100.      
  101.      }
  102.       
  103.      }
  104.    
  105.    
  106.   }
  107.   printf("将task[][]按字典排序结果/n【sort these lists in an ascending lexicographical order】/n");
  108.  for (int i=0;i<n;i++)
  109.  {
  110.     for(int j=0;j<5;j++)
  111.     {
  112.         if(src[i][j]!=-1)
  113.         printf("%d ",src[i][j]);}
  114.     printf("/n");
  115.  }
  116.     //-------------------------------------------------------------------------------------------------------------------------------
  117. }
  118. int maxitem(int a[4])//返回长度为4的数组中最大的项的下标
  119. {
  120.     int k=a[0]>a[1]?0:1;
  121.     k= a[k]>a[2]?k:2;
  122.     return a[k]>a[3]?k:3;
  123. }
  124. int minitem(int a[4])//返回长度为4的数组中最小项的下标【如果有最小项有相等的情况,则小下标优先】
  125. {
  126.     int k=a[3]<a[2]?3:2;
  127.     k=a[k]<a[1]?k:1;
  128.     return a[k]<a[0]?k:0;
  129. }
  130. void main()
  131. {
  132.     //定义任务图,采用二维数组表示所有节点间的关系,例如t1节点指向t2则node[1][2] = 4,而node[1][9] = 0,因为t1没有指向t9的路经
  133.     //由于没有节点t0则T[0][i]全部为0。
  134.     int i;
  135.     int j;
  136.     int node[10][10] = {0};
  137.     node[1][2] = 4; 
  138.     node[1][3] = 1;
  139.     node[1][4] = 1;
  140.     node[1][5] = 1;
  141.     node[1][7] = 10;
  142.     node[2][6] = 1;
  143.     node[2][7] = 1;
  144.   
  145.     node[3][8] = 1;
  146.     node[4][8] = 1;
  147.     
  148.     node[6][9] = 5;
  149.     
  150.     node[7][9] = 6;
  151.     
  152.     node[8][9] = 5;
  153. /********************************************************************************/  
  154.     //定义每一个节点的计算时间,忽略t[0]
  155.     int t[10];
  156.     t[1] = 2;
  157.     t[2] = 3;
  158.     t[3] = 3;
  159.     t[4] = 4;
  160.     t[5] = 5;
  161.     t[6] = 4;
  162.     t[7] = 4;
  163.     t[8] = 4;
  164.     t[9] = 1;
  165. /********************************************************************************/  
  166.     //定义每一个节点的前导节点,例如节点8的前导节点为节点3、4,则diaodao[8][3] = 1,diandao[8][4] = 1
  167.     int qiandao[10][10] = {0};
  168.     qiandao[2][1] = 1;
  169.     qiandao[3][1] = 1;
  170.     qiandao[4][1] = 1;
  171.     qiandao[5][1] = 1;
  172.     qiandao[7][1] = 1;
  173.     qiandao[6][2] = 1;
  174.     qiandao[7][2] = 1;
  175.     qiandao[8][3] = 1;
  176.     qiandao[8][4] = 1;
  177.     qiandao[9][6] = 1;
  178.     qiandao[9][7] = 1;
  179.     qiandao[9][8] = 1;
  180.         
  181. /********************************************************************************/  
  182.     //计算b_level值,忽略b_level[0],由于事先已知头节点和尾节点(在本例中头节点为t1,尾节点为t5和t9)
  183.     //所以从尾节点倒序计算每一个节点的b_level值。
  184.     int b_level[10] = {0};  
  185.     //尾节点b_level值为0
  186.     b_level[5] = 5;
  187.     b_level[9] = 1; 
  188.     //指示是否已经计算过b_level值
  189.     int b_cal[10] = {0};
  190.     b_cal[5] = 1;
  191.     b_cal[9] = 1;
  192.     int b_level_max, temp1, temp2, temp;
  193.     //从节点t1开始计算b_level值
  194.     
  195.     //从尾节点9开始倒序计算b_level值,直到头节点1结束,由于最多有三次向前计算的机会,所以算三次就结束。
  196.     for(int i = 1; i < 10; i++) 
  197.         if(qiandao[9][i] == 1)  
  198.             b_level[i] = t[9] + t[i] + node[i][9];  
  199.     //计算倒数第二排节点
  200.     for(int i = 1; i < 9; i++)
  201.     {
  202.         if(b_level[i] != 0)
  203.         {
  204.             for(int j = 1; j < 10; j++)
  205.                 if(qiandao[i][j] == 1)
  206.                 {
  207.                     temp1 = t[j] + node[j][i] + b_level[i];
  208.                     if(temp1 > b_level[j])
  209.                         b_level[j] = temp1;
  210.                 }
  211.         }
  212.     }   
  213.     //最后计算t1节点 
  214.     for(int i = 1; i < 10; i++)
  215.     {
  216.         if(qiandao[i][1] == 1)
  217.         {
  218.             temp1 = t[1] + node[1][i] + b_level[i];
  219.             if(temp1 > b_level[1])
  220.                 b_level[1] = temp1;
  221.         }
  222.         
  223.     }
  224.     
  225. /********************************************************************************/  
  226. /********************************************************************************/  
  227.     //计算b_level值,忽略b_level[0],由于事先已知头节点和尾节点(在本例中头节点为t1,尾节点为t5和t9)
  228.     //所以从尾节点倒序计算每一个节点的b_level值。
  229.     int SBL[10] = {0};  
  230.     //尾节点b_level值为0
  231.     SBL[5] = 5;
  232.     SBL[9] = 1; 
  233.     //指示是否已经计算过b_level值
  234.     int SBL_cal[10] = {0};
  235.     SBL_cal[5] = 1;
  236.     SBL_cal[9] = 1;
  237.     int SBL_max, SBL_temp1, SBL_temp2, SBL_temp;
  238.     //从节点t1开始计算b_level值
  239.     
  240.     //从尾节点9开始倒序计算b_level值,直到头节点1结束,由于最多有三次向前计算的机会,所以算三次就结束。
  241.     for(int i = 1; i < 10; i++) 
  242.         if(qiandao[9][i] == 1)  
  243.             SBL[i] =t[9] + t[i] ;       
  244.     //计算倒数第二排节点
  245.     for(int i = 1; i < 9; i++)
  246.     {
  247.         if(SBL[i] != 0)
  248.         {
  249.             for(int j = 1; j < 10; j++)
  250.                 if(qiandao[i][j] == 1)
  251.                 {
  252.                     SBL_temp1 = t[j]  + SBL[i];
  253.                     if(SBL_temp1 > SBL[j])
  254.                         SBL[j] = SBL_temp1;
  255.                 }
  256.         }
  257.     }   
  258.     //最后计算t1节点 
  259.     for(int i = 1; i < 10; i++)
  260.     {
  261.         if(qiandao[i][1] == 1)
  262.         {
  263.             SBL_temp1 =t[1]  + SBL[i];
  264.             if(SBL_temp1 > SBL[1])
  265.                 SBL[1] = SBL_temp1;
  266.         }
  267.         
  268.     }
  269.     
  270. //计算每个节点的前向大数
  271.     int qian_max[10] = {0};
  272.     for(i = 1; i < 10; i++)
  273.     {
  274.             temp1 = 0;
  275.             for(int j = 0; j < 10; j++)
  276.             {
  277.                 if(node[j][i] > qian_max[i])    
  278.                     qian_max[i] = node[j][i];
  279.             }           
  280.             
  281.     }
  282. /
  283. //计算节点t1到其他所有节点的最大路径值
  284.     int t_level[10] = {0};
  285.     for(i = 1; i < 10; i++)
  286.     {
  287.         if(node[1][i] != 0)
  288.         {
  289.             t_level[i] = t[1] + node[1][i];             
  290.             if(t_level[i] < temp1)
  291.                 t_level[i] = temp1;
  292.             for(int j = 1; j < 10; j++)
  293.             {
  294.                 if(node[i][j] != 0) 
  295.                 {
  296.                     t_level[j] = t_level[i] + node[i][j]+t[i];              
  297.                     for(int k = 1; k < 10; k++)
  298.                     {
  299.                         if(node[j][k] != 0) 
  300.                             t_level[k] = t_level[j] + node[j][k];
  301.                         
  302.                     }
  303.                 }
  304.             }   
  305.         }
  306.     }
  307.     for(i = 1; i < 10; i++)
  308.         printf("t_level[%d] = %d/n", i, t_level[i]);
  309.         
  310.         
  311.         for(i = 1; i < 10; i++)
  312.         printf("sbl[%d] = %d/n", i, SBL[i]);
  313.     for(i = 1; i < 10; i++)
  314.         printf("qian_max[%d] = %d/n", i, qian_max[i]);
  315.     for(i = 1; i < 10; i++) 
  316.     {printf("b_level[%d]: %d/n ",i, b_level[i]);}
  317.     
  318.     
  319.     
  320. //--这个部分为MCP算法的核心,得出就绪列表【开始】--------------------------------------------------------------------------------------------------------------------
  321. //(1)计算每个节点的ALAP时间
  322. //(2)对每个节点,创建它本身的以及它的所有子结点的ALAP时间按降序构成的表
  323. //(3)将这些表按字典排序,并按这个顺序创建一个节点表
  324.     int alap[10]={-1};//-1表示未赋值时的默认无效值【因为ALAP值不会是负数】
  325.     for(i=1;i<10;i++)
  326.     {
  327.     alap[i]=b_level[1]-b_level[i];//各个节点的ALAP值=关键路径长度-该节点的b_level值
  328.     }
  329. for(i = 1; i < 10; i++) 
  330.     {printf("alap[%d]: %d/n ",i, alap[i]);}
  331. //---section1[start]-----------------------------------------------------------------------------------------------------------
  332.     int task[9][5];
  333.     for(int i=0;i<9;i++)
  334.         for(int j=0;j<5;j++)
  335.             task[i][j]=-1;//task[][]中赋初值为-1,表示无效值(因为ALAP值不会是负数)
  336.    for(int i=1;i<10;i++)//找每个节点的子结点
  337.    {
  338.       int k=1;
  339.        task[i-1][0]=alap[i];//对每个节点,创建它本身的以及它的所有子结点的ALAP时间按降序构成的表【创建本身】
  340.        for(int j=1;j<10;j++)
  341.       
  342.           if(node[i][j]!=0)//找到一个子节点
  343.           {
  344.           task[i-1][k++]=alap[j];//对每个节点,创建它本身的以及它的所有子结点的ALAP时间按降序构成的表【创建所有子结点】
  345.           }
  346.    
  347.    }
  348. printf("以下是MCP算法得出的任务调度结果:/n【以节点的ALAP值以及该节点的子节点ALAP值 构造的task[][]为:】/n");
  349. for (int i=0;i<9;i++)
  350.  {
  351.     for(int j=0;j<5;j++)
  352.     {
  353.         if(task[i][j]!=-1)
  354.         printf("%d ",task[i][j]);}
  355.     printf("/n");
  356.  }
  357. //section1执行完后,task[][]在该默认例子中为{{0,8,9,8,18},{8,13,12},{9,13},{8,13},{18},{13,22},{12,22},{13,22},{22}}
  358. //----section1[end]------------------------------------------------------------------------------------------------------------
  359. //----section2[start]------------------------------------------------------------------------------------------------------------
  360. for(int i=0;i<10;i++)
  361. {
  362. BubbleSort(task[0]);
  363. }
  364. printf("【对task[][]中所有子结点的ALAP时间按降序排序后的结果:】/n");
  365. for (int i=0;i<9;i++)
  366.  {
  367.     for(int j=0;j<5;j++)
  368.     {
  369.         if(task[i][j]!=-1)
  370.         printf("%d ",task[i][j]);}
  371.     printf("/n");
  372.  }
  373. //section2执行完后,task[][]在该默认例子中为{{0,18,9,8,8},{8,13,12},{9,13},{8,13},{18},{13,22},{12,22},{13,22},{22}}
  374. //作用是除task[i][0]元素外,将task[i][1...]按降序排列
  375. //----section2[end]------------------------------------------------------------------------------------------------------------
  376. int seq[10]={0,1,2,3,4,5,6,7,8,9};
  377. DictionarySort(task,seq);//将task[][]字典排序
  378. //--这个部分为MCP算法的核心,得出就绪列表【结束】----------------------------------------------------------------------------------------------------------------------------------
  379. ISH
  380.     
  381.     
  382. int pe[10]={-1,-1,-1,-1,-1,-1,-1,-1,-1,-1};//记录任务的开始时间,如pe[i]=x表示i任务开始执行时间为x
  383. int time[4]={0,0,0,0};                     //记录机器当前时间,其值为当前分配的任务开始时间+任务的执行时间
  384. int tdispatched[10]={-1,-1,-1,-1,-1,-1,-1,-1,-1,-1};//记录任务的分配情况,如tdispatched[i]=x表示i任务被分配到x号机器上
  385. printf("【调度顺序:】/n");
  386. for(i=1;i<10;i++)  //从9个任务里面,按sbl的降序取值
  387. {
  388. int maxid=seq[i];
  389. printf("%d-",seq[i]);
  390. //=========================================================================================分配maxid开始
  391. if(maxid==1){
  392.         
  393.         pe[maxid]=0;
  394.         tdispatched[maxid]=0;
  395.         time[0]+=t[maxid];//0号机器被用掉的时间为该任务的运行耗时
  396.     
  397.     
  398.     
  399.     }else{
  400.         
  401.         
  402.         
  403.           
  404.         int early[4]={-1,-1,-1,-1};//记录一个任务如果分配到0-3号机器能够最早执行的时间
  405.     
  406. int a[4]={-1,-1,-1,-1};//记录由各个父节点产生的开始时间限制(包括同机器无通信时间情况与非同机器有通信时间情况)    
  407.     for(int machid=0;machid<4;machid++)//往4台机器上分
  408.     {
  409.   int j=1;
  410.   a[0]=-1;
  411.   a[1]=-1;
  412.   a[2]=-1;
  413.   a[3]=-1;
  414.   
  415.   //记录由各个父节点产生的开始时间限制(包括同机器无通信时间情况与非同机器有通信时间情况)
  416. //---------------------------------------------------------------------------------------------------------------trymap----
  417. a[0]=time[machid];//记录当前机器的当前时间,如果任务被分配当前机器,任务必须在该时间以后执行
  418. for(int i=1;i<10;i++)//遍历 找到maxid节点的前驱
  419.    {
  420.     if(node[i][maxid]!=0)//找到一个前驱,node[i][maxid]的值为节点maxid与节点i的通信时间
  421.       {
  422.          
  423.           
  424.           if(tdispatched[i]==machid)//如果前驱节点i与该节点maxid都被分到了同一个机器
  425.            {
  426.                
  427.                
  428.                a[j]=time[machid];//maxid任务开始时间紧接着该机器的当前时间
  429.                
  430.                j++;
  431.             }else{//如果前驱节点与该节点没有分配到同一个机器
  432.                a[j]=pe[i]+t[i]+node[i][maxid];//则maxid任务开始时间为前驱任务节点完成时间+与之通信的时间
  433.               
  434.                j++;
  435.             }
  436.     
  437.       }
  438.    }//for完成后,j为 maxid 节点 前驱的个数,a[0....(j-1)]为对应前驱节点产生的开始时间,其中最大的为必须满足的
  439.   
  440. early[machid]=a[maxitem(a)];
  441. //----------------------------------------------------------------------------------------------------------------trymap---
  442.     }   //得到early[0...3],里面最小的为选中的分配方案(开始时间最早)
  443.         
  444.     int fangan=minitem(early);//记录应该分配到哪台机器(开始时间最早)
  445.     tdispatched[maxid]=fangan;//tdispatched[maxid]记录maxid任务被分配到哪台机器
  446.     pe[maxid]=early[fangan];  //pe[maxid]记录maxid任务的开始时间
  447.     time[fangan]=pe[maxid]+t[maxid];
  448.     
  449.     
  450.     
  451.     }
  452. //==========================================================================================分配maxid结束
  453. }
  454. printf("/n【pe[i]表示i任务节点开始的执行时间】/n");
  455. for(i = 1; i < 10; i++) 
  456.     {printf("pe[%d]: %d/n ",i, pe[i]);}
  457. printf("【i任务节点被分配到编号为tdispatched[i]的机器】/n");
  458. for(i = 1; i < 10; i++) 
  459.     {printf("tdispatched[%d]: %d/n ",i, tdispatched[i]);}
  460. printf("【总共耗费的时间】");
  461.     {printf("%d/n ",time[maxitem(time)]);}
  462.     
  463.     
  464. }

 

 

 

 

  1. /*************************************************************************************************/
  2. //dls
  3. /*************************************************************************************************/
  4. #include "stdio.h"
  5. bool isallfinished(int *p)//判断是否所有节点都已调度完毕
  6. {
  7. for(int i=1;i<10;i++)
  8. {
  9.     if(p[i]!=1){return false;}
  10. }
  11. return true;
  12. }
  13. int maxitem(int a[4])//返回长度为4的数组中最大的项的下标
  14. {
  15.     int k=a[0]>a[1]?0:1;
  16.     k= a[k]>a[2]?k:2;
  17.     return a[k]>a[3]?k:3;
  18. }
  19. int minitem(int a[4])//返回长度为4的数组中最小项的下标【如果有最小项有相等的情况,则小下标优先】
  20. {
  21.     int k=a[3]<a[2]?3:2;
  22.     k=a[k]<a[1]?k:1;
  23.     return a[k]<a[0]?k:0;
  24. }
  25. void main()
  26. {
  27.     //定义任务图,采用二维数组表示所有节点间的关系,例如t1节点指向t2则node[1][2] = 4,而node[1][9] = 0,因为t1没有指向t9的路经
  28.     //由于没有节点t0则T[0][i]全部为0。
  29.     int i;
  30.     int j;
  31.     int node[10][10] = {0};
  32.     node[1][2] = 4; 
  33.     node[1][3] = 1;
  34.     node[1][4] = 1;
  35.     node[1][5] = 1;
  36.     node[1][7] = 10;
  37.     node[2][6] = 1;
  38.     node[2][7] = 1;
  39.   
  40.     node[3][8] = 1;
  41.     node[4][8] = 1;
  42.     
  43.     node[6][9] = 5;
  44.     
  45.     node[7][9] = 6;
  46.     
  47.     node[8][9] = 5;
  48. /********************************************************************************/  
  49.     //定义每一个节点的计算时间,忽略t[0]
  50.     int t[10];
  51.     t[1] = 2;
  52.     t[2] = 3;
  53.     t[3] = 3;
  54.     t[4] = 4;
  55.     t[5] = 5;
  56.     t[6] = 4;
  57.     t[7] = 4;
  58.     t[8] = 4;
  59.     t[9] = 1;
  60. /********************************************************************************/  
  61.     //定义每一个节点的前导节点,例如节点8的前导节点为节点3、4,则diaodao[8][3] = 1,diandao[8][4] = 1
  62.     int qiandao[10][10] = {0};
  63.     qiandao[2][1] = 1;
  64.     qiandao[3][1] = 1;
  65.     qiandao[4][1] = 1;
  66.     qiandao[5][1] = 1;
  67.     qiandao[7][1] = 1;
  68.     qiandao[6][2] = 1;
  69.     qiandao[7][2] = 1;
  70.     qiandao[8][3] = 1;
  71.     qiandao[8][4] = 1;
  72.     qiandao[9][6] = 1;
  73.     qiandao[9][7] = 1;
  74.     qiandao[9][8] = 1;
  75.         
  76. /********************************************************************************/  
  77.     //计算b_level值,忽略b_level[0],由于事先已知头节点和尾节点(在本例中头节点为t1,尾节点为t5和t9)
  78.     //所以从尾节点倒序计算每一个节点的b_level值。
  79.     int b_level[10] = {0};  
  80.     //尾节点b_level值为0
  81.     b_level[5] = 5;
  82.     b_level[9] = 1; 
  83.     //指示是否已经计算过b_level值
  84.     int b_cal[10] = {0};
  85.     b_cal[5] = 1;
  86.     b_cal[9] = 1;
  87.     int b_level_max, temp1, temp2, temp;
  88.     //从节点t1开始计算b_level值
  89.     
  90.     //从尾节点9开始倒序计算b_level值,直到头节点1结束,由于最多有三次向前计算的机会,所以算三次就结束。
  91.     for(int i = 1; i < 10; i++) 
  92.         if(qiandao[9][i] == 1)  
  93.             b_level[i] = t[9] + t[i] + node[i][9];  
  94.     //计算倒数第二排节点
  95.     for(int i = 1; i < 9; i++)
  96.     {
  97.         if(b_level[i] != 0)
  98.         {
  99.             for(int j = 1; j < 10; j++)
  100.                 if(qiandao[i][j] == 1)
  101.                 {
  102.                     temp1 = t[j] + node[j][i] + b_level[i];
  103.                     if(temp1 > b_level[j])
  104.                         b_level[j] = temp1;
  105.                 }
  106.         }
  107.     }   
  108.     //最后计算t1节点 
  109.     for(int i = 1; i < 10; i++)
  110.     {
  111.         if(qiandao[i][1] == 1)
  112.         {
  113.             temp1 = t[1] + node[1][i] + b_level[i];
  114.             if(temp1 > b_level[1])
  115.                 b_level[1] = temp1;
  116.         }
  117.         
  118.     }
  119.     
  120. /********************************************************************************/  
  121. /********************************************************************************/  
  122.     //计算b_level值,忽略b_level[0],由于事先已知头节点和尾节点(在本例中头节点为t1,尾节点为t5和t9)
  123.     //所以从尾节点倒序计算每一个节点的b_level值。
  124.     int SBL[10] = {0};  
  125.     //尾节点b_level值为0
  126.     SBL[5] = 5;
  127.     SBL[9] = 1; 
  128.     //指示是否已经计算过b_level值
  129.     int SBL_cal[10] = {0};
  130.     SBL_cal[5] = 1;
  131.     SBL_cal[9] = 1;
  132.     int SBL_max, SBL_temp1, SBL_temp2, SBL_temp;
  133.     //从节点t1开始计算b_level值
  134.     
  135.     //从尾节点9开始倒序计算b_level值,直到头节点1结束,由于最多有三次向前计算的机会,所以算三次就结束。
  136.     for(int i = 1; i < 10; i++) 
  137.         if(qiandao[9][i] == 1)  
  138.             SBL[i] =t[9] + t[i] ;       
  139.     //计算倒数第二排节点
  140.     for(int i = 1; i < 9; i++)
  141.     {
  142.         if(SBL[i] != 0)
  143.         {
  144.             for(int j = 1; j < 10; j++)
  145.                 if(qiandao[i][j] == 1)
  146.                 {
  147.                     SBL_temp1 = t[j]  + SBL[i];
  148.                     if(SBL_temp1 > SBL[j])
  149.                         SBL[j] = SBL_temp1;
  150.                 }
  151.         }
  152.     }   
  153.     //最后计算t1节点 
  154.     for(int i = 1; i < 10; i++)
  155.     {
  156.         if(qiandao[i][1] == 1)
  157.         {
  158.             SBL_temp1 =t[1]  + SBL[i];
  159.             if(SBL_temp1 > SBL[1])
  160.                 SBL[1] = SBL_temp1;
  161.         }
  162.         
  163.     }
  164.     
  165. //计算每个节点的前向大数
  166.     int qian_max[10] = {0};
  167.     for(i = 1; i < 10; i++)
  168.     {
  169.             temp1 = 0;
  170.             for(int j = 0; j < 10; j++)
  171.             {
  172.                 if(node[j][i] > qian_max[i])    
  173.                     qian_max[i] = node[j][i];
  174.             }           
  175.             
  176.     }
  177. /
  178. //计算节点t1到其他所有节点的最大路径值
  179.     int t_level[10] = {0};
  180.     for(i = 1; i < 10; i++)
  181.     {
  182.         if(node[1][i] != 0)
  183.         {
  184.             t_level[i] = t[1] + node[1][i];             
  185.             if(t_level[i] < temp1)
  186.                 t_level[i] = temp1;
  187.             for(int j = 1; j < 10; j++)
  188.             {
  189.                 if(node[i][j] != 0) 
  190.                 {
  191.                     t_level[j] = t_level[i] + node[i][j]+t[i];              
  192.                     for(int k = 1; k < 10; k++)
  193.                     {
  194.                         if(node[j][k] != 0) 
  195.                             t_level[k] = t_level[j] + node[j][k];
  196.                         
  197.                     }
  198.                 }
  199.             }   
  200.         }
  201.     }
  202.     for(i = 1; i < 10; i++)
  203.         printf("t_level[%d] = %d/n", i, t_level[i]);
  204.         
  205.         
  206.         for(i = 1; i < 10; i++)
  207.         printf("sbl[%d] = %d/n", i, SBL[i]);
  208.     for(i = 1; i < 10; i++)
  209.         printf("qian_max[%d] = %d/n", i, qian_max[i]);
  210.     for(i = 1; i < 10; i++) 
  211.     {printf("b_level[%d]: %d/n ",i, b_level[i]);}
  212.     
  213.     
  214. int seq[10]={0,1,4,3,5,2,8,6,7,9};
  215. int pe[10]={-1,-1,-1,-1,-1,-1,-1,-1,-1,-1};//记录任务的开始时间,如pe[i]=x表示i任务开始执行时间为x
  216. int time[4]={0,0,0,0};                     //记录机器当前时间,其值为当前分配的任务开始时间+任务的执行时间
  217. int tdispatched[10]={-1,-1,-1,-1,-1,-1,-1,-1,-1,-1};//记录任务的分配情况,如tdispatched[i]=x表示i任务被分配到x号机器上
  218.     int isready[10]={0,1,1,1,1,1,1,1,1,1};//ready[i]=0表示i 任务未就绪,ready[i]=1表示任务i就绪(注1:该表的初值在下面的FOR中初始化;注2:所谓就绪指任务的父节点已执行完)
  219.     int isfinish[10]={0};//初始状态所有节点都未执行
  220.     
  221.     
  222.     
  223. while(!isallfinished(isfinish))//反复调度直到所有节点都执行完毕
  224. {
  225.     
  226.     
  227.     int trystarttime[10]={-1,-1,-1,-1,-1,-1,-1,-1,-1,-1};
  228.     int trymap[10]={-1,-1,-1,-1,-1,-1,-1,-1,-1,-1};
  229.     
  230.     int early[4]={-1,-1,-1,-1};//记录一个任务如果分配到0-3号机器能够最早执行的时间
  231.     int selectedid;
  232.     int dl[10]={-1024,-1024,-1024,-1024,-1024,-1024,-1024,-1024,-1024,-1024};
  233.     for(int i=1;i<10;i++)//更新就绪列表
  234.     {
  235.       
  236.       bool flag=true;
  237.       for(int j=1;j<10;j++)
  238.       {
  239.          
  240.           if(node[j][i]!=0)
  241.           {
  242.            if(isfinish[j]==0)//只要有一个父节点未执行完成,则该节点状态为未就绪
  243.            {
  244.             flag=false;
  245.                
  246.            }
  247.           }
  248.        }
  249.       if(flag&&(isfinish[i]==0))//当该节点所有父节点都调度,并且该节点未被调度时,该节点标记为就绪节点
  250.       {
  251.       isready[i]=1;
  252.       }else
  253.       {
  254.       isready[i]=0;
  255.       }
  256.     }
  257.     
  258. for(int i=1;i<10;i++)  
  259. {
  260. if(isready[i]==1)
  261. {
  262. selectedid=i;
  263. }else{continue;}
  264. //=========================================================================================分配selectedid开始
  265. if(selectedid==1){
  266.         
  267.         
  268.     
  269.         
  270.         trymap[selectedid]=0;
  271.         trystarttime[selectedid]=0;
  272.     
  273.     }else{
  274.         
  275.         
  276.         
  277.           
  278.         
  279.     
  280. int a[4]={-1,-1,-1,-1};//记录由各个父节点产生的开始时间限制(包括同机器无通信时间情况与非同机器有通信时间情况)    
  281.     for(int machid=0;machid<4;machid++)//往4台机器上分
  282.     {
  283.   int j=1;
  284.   a[0]=-1;
  285.   a[1]=-1;
  286.   a[2]=-1;
  287.   a[3]=-1;
  288.   
  289.   //记录由各个父节点产生的开始时间限制(包括同机器无通信时间情况与非同机器有通信时间情况)
  290. //---------------------------------------------------------------------------------------------------------------trymap----
  291. a[0]=time[machid];//记录当前机器的当前时间,如果任务被分配当前机器,任务必须在该时间以后执行
  292. for(int i=1;i<10;i++)//遍历 找到selectedid节点的前驱
  293.    {
  294.     if(node[i][selectedid]!=0)//找到一个前驱,node[i][selectedid]的值为节点selectedid与节点i的通信时间
  295.       {
  296.          
  297.           
  298.           if(tdispatched[i]==machid)//如果前驱节点i与该节点selectedid都被分到了同一个机器
  299.            {
  300.                
  301.                
  302.                a[j]=time[machid];//selectedid任务开始时间紧接着该机器的当前时间
  303.                
  304.                j++;
  305.             }else{//如果前驱节点与该节点没有分配到同一个机器
  306.                a[j]=pe[i]+t[i]+node[i][selectedid];//则selectedid任务开始时间为前驱任务节点完成时间+与之通信的时间
  307.               
  308.                j++;
  309.             }
  310.     
  311.       }
  312.    }//for完成后,j为 selectedid 节点 前驱的个数,a[0....(j-1)]为对应前驱节点产生的开始时间,其中最大的为必须满足的
  313.   
  314. early[machid]=a[maxitem(a)];
  315. //----------------------------------------------------------------------------------------------------------------trymap---
  316.     }   //得到early[0...3],里面最小的为选中的分配方案(开始时间最早)
  317.     
  318.     
  319.   
  320.    trymap[selectedid]=minitem(early);
  321.    trystarttime[selectedid]=early[minitem(early)];
  322.     
  323.     
  324.     }//(else)
  325. //==========================================================================================分配selectedid结束
  326. }//(for)就绪的节点都求到了最早开始时间,记录在trystarttime[]里,trymap[]则记录了分配方案,trydl[]里记录了每个就绪节点的最大DL
  327. int k1;
  328. //====================================================================================================取最大DL
  329. for(int i=1;i<10;i++)//求出所有就绪节点的DL值,节点DL值=节点SBL值-节点最早开始时间
  330. {
  331.   if(isready[i]==1)
  332.     dl[i]=SBL[i]-trystarttime[i];
  333. }
  334.     int k=dl[9]>dl[8]?9:8;
  335.     k=dl[k]>dl[7]?k:7;
  336.     k=dl[k]>dl[6]?k:6;
  337.     k=dl[k]>dl[5]?k:5;
  338.     k=dl[k]>dl[4]?k:4;
  339.     k=dl[k]>dl[3]?k:3;
  340.     k=dl[k]>dl[2]?k:2;
  341.     k=dl[k]>dl[1]?k:1;//取具有最大DL值的节点
  342. k1=k;//k1的值为所有就绪节点中,具有最大DL节点-处理机对的编号,
  343. //====================================================================================================
  344. int fangan=trymap[k1];//记录应该分配到哪台机器(开始时间最早)
  345.     tdispatched[k1]=fangan;//tdispatched[selectedid]记录selectedid任务被分配到哪台机器
  346.     pe[k1]=trystarttime[k1];  //pe[selectedid]记录selectedid任务的开始时间
  347.     time[fangan]=pe[k1]+t[k1];//该机器(fangan)执行该任务(k1)后的时间
  348.     isfinish[k1]=1;//将该任务标记为已调度
  349. }//while(!isallfinished(isfinish))这个循环执行到所有任务节点都被完成
  350.     
  351.     
  352. printf("/n【pe[i]表示i任务节点开始的执行时间】/n");
  353. for(i = 1; i < 10; i++) 
  354.     {printf("pe[%d]: %d/n ",i, pe[i]);}
  355. printf("【i任务节点被分配到编号为tdispatched[i]的机器】/n");
  356. for(i = 1; i < 10; i++) 
  357.     {printf("tdispatched[%d]: %d/n ",i, tdispatched[i]);}
  358. printf("【总共耗费的时间】");
  359.     {printf("%d/n ",time[maxitem(time)]);}
  360. }

 

 

 

 

 

 

  1. /*************************************************************************************************/
  2. //ETF
  3. /*************************************************************************************************/
  4. #include "stdio.h"
  5. bool isallfinished(int *p)//判断是否所有节点都已调度完毕
  6. {
  7. for(int i=1;i<10;i++)
  8. {
  9.     if(p[i]!=1){return false;}
  10. }
  11. return true;
  12. }
  13. int maxitem(int a[4])//返回长度为4的数组中最大的项的下标
  14. {
  15.     int k=a[0]>a[1]?0:1;
  16.     k= a[k]>a[2]?k:2;
  17.     return a[k]>a[3]?k:3;
  18. }
  19. int minitem(int a[4])//返回长度为4的数组中最小项的下标【如果有最小项有相等的情况,则小下标优先】
  20. {
  21.     int k=a[3]<a[2]?3:2;
  22.     k=a[k]<a[1]?k:1;
  23.     return a[k]<a[0]?k:0;
  24. }
  25. void main()
  26. {
  27.     //定义任务图,采用二维数组表示所有节点间的关系,例如t1节点指向t2则node[1][2] = 4,而node[1][9] = 0,因为t1没有指向t9的路经
  28.     //由于没有节点t0则T[0][i]全部为0。
  29.     int i;
  30.     int j;
  31.     int node[10][10] = {0};
  32.     node[1][2] = 4; 
  33.     node[1][3] = 1;
  34.     node[1][4] = 1;
  35.     node[1][5] = 1;
  36.     node[1][7] = 10;
  37.     node[2][6] = 1;
  38.     node[2][7] = 1;
  39.   
  40.     node[3][8] = 1;
  41.     node[4][8] = 1;
  42.     
  43.     node[6][9] = 5;
  44.     
  45.     node[7][9] = 6;
  46.     
  47.     node[8][9] = 5;
  48. /********************************************************************************/  
  49.     //定义每一个节点的计算时间,忽略t[0]
  50.     int t[10];
  51.     t[1] = 2;
  52.     t[2] = 3;
  53.     t[3] = 3;
  54.     t[4] = 4;
  55.     t[5] = 5;
  56.     t[6] = 4;
  57.     t[7] = 4;
  58.     t[8] = 4;
  59.     t[9] = 1;
  60. /********************************************************************************/  
  61.     //定义每一个节点的前导节点,例如节点8的前导节点为节点3、4,则diaodao[8][3] = 1,diandao[8][4] = 1
  62.     int qiandao[10][10] = {0};
  63.     qiandao[2][1] = 1;
  64.     qiandao[3][1] = 1;
  65.     qiandao[4][1] = 1;
  66.     qiandao[5][1] = 1;
  67.     qiandao[7][1] = 1;
  68.     qiandao[6][2] = 1;
  69.     qiandao[7][2] = 1;
  70.     qiandao[8][3] = 1;
  71.     qiandao[8][4] = 1;
  72.     qiandao[9][6] = 1;
  73.     qiandao[9][7] = 1;
  74.     qiandao[9][8] = 1;
  75.         
  76. /********************************************************************************/  
  77.     //计算b_level值,忽略b_level[0],由于事先已知头节点和尾节点(在本例中头节点为t1,尾节点为t5和t9)
  78.     //所以从尾节点倒序计算每一个节点的b_level值。
  79.     int b_level[10] = {0};  
  80.     //尾节点b_level值为0
  81.     b_level[5] = 5;
  82.     b_level[9] = 1; 
  83.     //指示是否已经计算过b_level值
  84.     int b_cal[10] = {0};
  85.     b_cal[5] = 1;
  86.     b_cal[9] = 1;
  87.     int b_level_max, temp1, temp2, temp;
  88.     //从节点t1开始计算b_level值
  89.     
  90.     //从尾节点9开始倒序计算b_level值,直到头节点1结束,由于最多有三次向前计算的机会,所以算三次就结束。
  91.     for(int i = 1; i < 10; i++) 
  92.         if(qiandao[9][i] == 1)  
  93.             b_level[i] = t[9] + t[i] + node[i][9];  
  94.     //计算倒数第二排节点
  95.     for(int i = 1; i < 9; i++)
  96.     {
  97.         if(b_level[i] != 0)
  98.         {
  99.             for(int j = 1; j < 10; j++)
  100.                 if(qiandao[i][j] == 1)
  101.                 {
  102.                     temp1 = t[j] + node[j][i] + b_level[i];
  103.                     if(temp1 > b_level[j])
  104.                         b_level[j] = temp1;
  105.                 }
  106.         }
  107.     }   
  108.     //最后计算t1节点 
  109.     for(int i = 1; i < 10; i++)
  110.     {
  111.         if(qiandao[i][1] == 1)
  112.         {
  113.             temp1 = t[1] + node[1][i] + b_level[i];
  114.             if(temp1 > b_level[1])
  115.                 b_level[1] = temp1;
  116.         }
  117.         
  118.     }
  119.     
  120. /********************************************************************************/  
  121. /********************************************************************************/  
  122.     //计算b_level值,忽略b_level[0],由于事先已知头节点和尾节点(在本例中头节点为t1,尾节点为t5和t9)
  123.     //所以从尾节点倒序计算每一个节点的b_level值。
  124.     int SBL[10] = {0};  
  125.     //尾节点b_level值为0
  126.     SBL[5] = 5;
  127.     SBL[9] = 1; 
  128.     //指示是否已经计算过b_level值
  129.     int SBL_cal[10] = {0};
  130.     SBL_cal[5] = 1;
  131.     SBL_cal[9] = 1;
  132.     int SBL_max, SBL_temp1, SBL_temp2, SBL_temp;
  133.     //从节点t1开始计算b_level值
  134.     
  135.     //从尾节点9开始倒序计算b_level值,直到头节点1结束,由于最多有三次向前计算的机会,所以算三次就结束。
  136.     for(int i = 1; i < 10; i++) 
  137.         if(qiandao[9][i] == 1)  
  138.             SBL[i] =t[9] + t[i] ;       
  139.     //计算倒数第二排节点
  140.     for(int i = 1; i < 9; i++)
  141.     {
  142.         if(SBL[i] != 0)
  143.         {
  144.             for(int j = 1; j < 10; j++)
  145.                 if(qiandao[i][j] == 1)
  146.                 {
  147.                     SBL_temp1 = t[j]  + SBL[i];
  148.                     if(SBL_temp1 > SBL[j])
  149.                         SBL[j] = SBL_temp1;
  150.                 }
  151.         }
  152.     }   
  153.     //最后计算t1节点 
  154.     for(int i = 1; i < 10; i++)
  155.     {
  156.         if(qiandao[i][1] == 1)
  157.         {
  158.             SBL_temp1 =t[1]  + SBL[i];
  159.             if(SBL_temp1 > SBL[1])
  160.                 SBL[1] = SBL_temp1;
  161.         }
  162.         
  163.     }
  164.     
  165. //计算每个节点的前向大数
  166.     int qian_max[10] = {0};
  167.     for(i = 1; i < 10; i++)
  168.     {
  169.             temp1 = 0;
  170.             for(int j = 0; j < 10; j++)
  171.             {
  172.                 if(node[j][i] > qian_max[i])    
  173.                     qian_max[i] = node[j][i];
  174.             }           
  175.             
  176.     }
  177. /
  178. //计算节点t1到其他所有节点的最大路径值
  179.     int t_level[10] = {0};
  180.     for(i = 1; i < 10; i++)
  181.     {
  182.         if(node[1][i] != 0)
  183.         {
  184.             t_level[i] = t[1] + node[1][i];             
  185.             if(t_level[i] < temp1)
  186.                 t_level[i] = temp1;
  187.             for(int j = 1; j < 10; j++)
  188.             {
  189.                 if(node[i][j] != 0) 
  190.                 {
  191.                     t_level[j] = t_level[i] + node[i][j]+t[i];              
  192.                     for(int k = 1; k < 10; k++)
  193.                     {
  194.                         if(node[j][k] != 0) 
  195.                             t_level[k] = t_level[j] + node[j][k];
  196.                         
  197.                     }
  198.                 }
  199.             }   
  200.         }
  201.     }
  202.     for(i = 1; i < 10; i++)
  203.         printf("t_level[%d] = %d/n", i, t_level[i]);
  204.         
  205.         
  206.         for(i = 1; i < 10; i++)
  207.         printf("sbl[%d] = %d/n", i, SBL[i]);
  208.     for(i = 1; i < 10; i++)
  209.         printf("qian_max[%d] = %d/n", i, qian_max[i]);
  210.     for(i = 1; i < 10; i++) 
  211.     {printf("b_level[%d]: %d/n ",i, b_level[i]);}
  212.     
  213.     
  214. int seq[10]={0,1,4,3,5,2,8,6,7,9};
  215. int pe[10]={-1,-1,-1,-1,-1,-1,-1,-1,-1,-1};//记录任务的开始时间,如pe[i]=x表示i任务开始执行时间为x
  216. int time[4]={0,0,0,0};                     //记录机器当前时间,其值为当前分配的任务开始时间+任务的执行时间
  217. int tdispatched[10]={-1,-1,-1,-1,-1,-1,-1,-1,-1,-1};//记录任务的分配情况,如tdispatched[i]=x表示i任务被分配到x号机器上
  218.     int isready[10]={0,1,1,1,1,1,1,1,1,1};//ready[i]=0表示i 任务未就绪,ready[i]=1表示任务i就绪(注1:该表的初值在下面的FOR中初始化;注2:所谓就绪指任务的父节点已执行完)
  219.     int isfinish[10]={0};//初始状态所有节点都未执行
  220.     
  221.     
  222.     
  223. while(!isallfinished(isfinish))//反复调度直到所有节点都执行完毕
  224. {
  225.     
  226.     
  227.     int trystarttime[10]={-1,-1,-1,-1,-1,-1,-1,-1,-1,-1};
  228.     int trymap[10]={-1,-1,-1,-1,-1,-1,-1,-1,-1,-1};
  229.     
  230.     int early[4]={-1,-1,-1,-1};//记录一个任务如果分配到0-3号机器能够最早执行的时间
  231.     int selectedid;
  232.     for(int i=1;i<10;i++)//更新就绪列表
  233.     {
  234.       
  235.       bool flag=true;
  236.       for(int j=1;j<10;j++)
  237.       {
  238.          
  239.           if(node[j][i]!=0)
  240.           {
  241.            if(isfinish[j]==0)//只要有一个父节点未执行完成,则该节点状态为未就绪
  242.            {
  243.             flag=false;
  244.                
  245.            }
  246.           }
  247.        }
  248.       if(flag&&(isfinish[i]==0))//如果所有父节点都完成,并且该节点未被调度,则该节点属于就绪节点
  249.       {
  250.       isready[i]=1;
  251.       }else
  252.       {
  253.       isready[i]=0;
  254.       }
  255.     }
  256.     
  257. for(int i=1;i<10;i++)  //遍历所有节点
  258. {
  259. if(isready[i]==1)//如果该节点属于就绪节点
  260. {
  261. selectedid=i;    //则对该节点进行考察
  262. }else{continue;}//否则取下一个节点
  263. //=========================================================================================分配selectedid开始
  264. if(selectedid==1){
  265.         
  266.         
  267.     
  268.         
  269.         trymap[selectedid]=0;
  270.         trystarttime[selectedid]=0;
  271.     
  272.     }else{
  273.         
  274.         
  275.         
  276.           
  277.         
  278.     
  279. int a[4]={-1,-1,-1,-1};//记录由各个父节点产生的开始时间限制(包括同机器无通信时间情况与非同机器有通信时间情况)    
  280.     for(int machid=0;machid<4;machid++)//往4台机器上分
  281.     {
  282.   int j=1;
  283.   a[0]=-1;
  284.   a[1]=-1;
  285.   a[2]=-1;
  286.   a[3]=-1;
  287.   
  288.   //记录由各个父节点产生的开始时间限制(包括同机器无通信时间情况与非同机器有通信时间情况)
  289. //---------------------------------------------------------------------------------------------------------------trymap----
  290. a[0]=time[machid];//记录当前机器的当前时间,如果任务被分配当前机器,任务必须在该时间以后执行
  291. for(int i=1;i<10;i++)//遍历 找到selectedid节点的前驱
  292.    {
  293.     if(node[i][selectedid]!=0)//找到一个前驱,node[i][selectedid]的值为节点selectedid与节点i的通信时间
  294.       {
  295.          
  296.           
  297.           if(tdispatched[i]==machid)//如果前驱节点i与该节点selectedid都被分到了同一个机器
  298.            {
  299.                
  300.                
  301.                a[j]=time[machid];//selectedid任务开始时间紧接着该机器的当前时间
  302.                
  303.                j++;
  304.             }else{//如果前驱节点与该节点没有分配到同一个机器
  305.                a[j]=pe[i]+t[i]+node[i][selectedid];//则selectedid任务开始时间为前驱任务节点完成时间+与之通信的时间
  306.               
  307.                j++;
  308.             }
  309.     
  310.       }
  311.    }//for完成后,j为 selectedid 节点 前驱的个数,a[0....(j-1)]为对应前驱节点产生的开始时间,其中最大的为必须满足的
  312.   
  313. early[machid]=a[maxitem(a)];
  314. //----------------------------------------------------------------------------------------------------------------trymap---
  315.     }   //得到early[0...3],里面最小的为选中的分配方案(开始时间最早)
  316.     
  317.     
  318.   
  319.    trymap[selectedid]=minitem(early);//使该节点(selectedid)有最早开始时间的机器
  320.    trystarttime[selectedid]=early[minitem(early)];//该节点(selectedid)的最早开始时间
  321.     
  322.     
  323.     }//(else)
  324. //==========================================================================================分配selectedid结束
  325. }//(for)就绪的节点都求到了最早开始时间,记录在trystarttime[]里,trymap[]则记录了分配方案,trydl[]里记录了每个就绪节点的最大DL
  326. int k1;
  327. //====================================================================================================取最早开始时间,如果时间一样 取小SBL
  328. for(int i=1;i<10;i++)
  329.     {
  330.     if(trystarttime[i]==-1) trystarttime[i]=1024;  //非就绪节点的开始时间设为无穷大(用1024代替)
  331.     
  332.     }
  333.     int k=trystarttime[9]<trystarttime[8]?9:8;
  334.     k=trystarttime[k]<trystarttime[7]?k:7;
  335.     k=trystarttime[k]<trystarttime[6]?k:6;
  336.     k=trystarttime[k]<trystarttime[5]?k:5;
  337.     k=trystarttime[k]<trystarttime[4]?k:4;
  338.     k=trystarttime[k]<trystarttime[3]?k:3;
  339.     k=trystarttime[k]<trystarttime[2]?k:2;
  340.     k=trystarttime[k]<trystarttime[1]?k:1;//取所有就绪节点中,具有最早开始时间的(存在这样的情况:几个节点同时具有最早开始时间)
  341.     for(int i=1;i<10;i++)//在几个节点同时具有最早开始时间的情况下,考察SBL,小SBL优先分配
  342.     {
  343.       if(trystarttime[k]==trystarttime[i])
  344.       {
  345.         if(SBL[k]<SBL[i])
  346.         {
  347.          k=i;
  348.         }
  349.       }
  350.     }
  351. k1=k;//k1的值为所有就绪节点中,选中准备分配的编号
  352. //====================================================================================================
  353. int fangan=trymap[k1];//记录应该分配到哪台机器(开始时间最早)
  354.     tdispatched[k1]=fangan;//tdispatched[selectedid]记录selectedid任务被分配到哪台机器
  355.     pe[k1]=trystarttime[k1];  //pe[selectedid]记录selectedid任务的开始时间
  356.     time[fangan]=pe[k1]+t[k1];//该机器(fangan)运行该任务(k1)后的时间
  357.     isfinish[k1]=1;//将该节点标记为已完成
  358. }//while(!isallfinished(isfinish))这个循环执行到所有任务节点都被完成
  359.     
  360.     
  361. printf("/n【pe[i]表示i任务节点开始的执行时间】/n");
  362. for(i = 1; i < 10; i++) 
  363.     {printf("pe[%d]: %d/n ",i, pe[i]);}
  364. printf("【i任务节点被分配到编号为tdispatched[i]的机器】/n");
  365. for(i = 1; i < 10; i++) 
  366.     {printf("tdispatched[%d]: %d/n ",i, tdispatched[i]);}
  367. printf("【总共耗费的时间】");
  368.     {printf("%d/n ",time[maxitem(time)]);}
  369. }

 

 

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值