图的更多相关算法-2(最短路径)

  
  
  
  
  1. void DispPath(int path[],int begin, int end)               //此函数是输出从begin节点到end节点最短的路径 
  2.     int ivar = 0; 
  3.     ivar = path[end]; 
  4.     cout<<"path is :"<<end<<' '
  5.     while(ivar != 0) 
  6.     {        
  7.         cout<<ivar<<' '
  8.         ivar = path[ivar]; 
  9.     } 
  10.     cout<<begin<<endl; 
  11.  
  12. /* 
  13.     此算法求节点距离其他节点的最短距离 
  14.     s[i] == 1数组表示节点i在 s中不再 U中 
  15.     dist[i]数组保存从begin到i节点的路径最小值 
  16.     path[i]数组保存的是路径中在i节点前面的节点的值 
  17.     思路: 
  18.     在dist数组中查找在s数组中最短的路径的值,把找到的节点当做中间节点, 如果通过中间节点的路径比直达的路径或原先的路径短就进行替换  
  19.     整个查找中间节点的操作是在dist数组中进行的 
  20.  
  21.   此算法与普利姆算法的区别是当找到中间节点后,此算法会由中间节点计算权的值,并进行相关的计算 
  22.   而普利姆算法则是利用中间的修正候选边 
  23.  
  24.   普利姆算法中的lowcost保存的是两个顶点之间的最短路径 
  25.   而狄更拉斯算法中的  dist数组保存的是初始顶点到其他顶点之间的最短路径,中间可以有节点存在即不是直达的节点 
  26. */ 
  27. void Dijkstra(MGraph mg, int begin) 
  28.     int loop1 = 0, count = 0, min = 1000, node = 0, path[MAXSIZE]; //path[i] 保存的是从begin 到i的路径中,位于 i前面的一个节点的值 
  29.     int s[MAXSIZE], dist[MAXSIZE];                              //s数组用以说明顶点的位置,当s[i] == 1的时候,说明顶点i在s中, 当s[i] == 0 的时候说明顶点在U中 
  30.                                                                 //dist[i]数组用以保存从顶点begin 到 顶点为i的最短距离 
  31.     for(loop1 = 0; loop1 < MAXSIZE; ++loop1) 
  32.         s[loop1] = 0; 
  33.     s[begin] = 1;    
  34.  
  35.     for(loop1 = 0; loop1 < mg.n; ++loop1) 
  36.     {    
  37.         dist[loop1] = mg.edges[begin][loop1];                   //dist[i] == 0的话,表示从 顶点begin 到 i节点不可到达 
  38.         if(mg.edges[begin][loop1] == 0)                         //此处为path赋初值,如果begin到节点loop1可直达,则path为0 代表这个路径的前一个节点是begin节点本身, 否则的话path初始化为-1 ,代表没有路径 
  39.         {   path[count] = -1; ++count; } 
  40.         else 
  41.         {   path[count] =  0; ++count; } 
  42.     } 
  43.     path[begin] = 0; 
  44.     count = 1; 
  45.  
  46.  
  47.     while(count < mg.n) 
  48.     { 
  49.         min = 1000; 
  50.         for(loop1 = 0; loop1 < mg.n; ++loop1)               //此处进行查找最短的路径,条件是节点在U中,并且从begin到节点必须可以到达 
  51.         { 
  52.             if(s[loop1] == 0 && dist[loop1] != 0 && min > dist[loop1]) 
  53.             { 
  54.                 min  = dist[loop1]; 
  55.                 node = loop1; 
  56.             } 
  57.         } 
  58.         cout<<"Visiting node  "<<node<<endl; 
  59.  
  60.         s[node] = 1;                                        //node为找到的节点 
  61.         for(loop1 = 1; loop1 < mg.n; ++loop1)               //此步骤是以找到的node节点为点,循环遍历与node直接相邻的量 
  62.         { 
  63.             if(node != loop1 && mg.edges[node][loop1] != 0)//条件是不遍历自己, 并且那个节点与node节点应该可直达 
  64.             { 
  65.                 if(dist[loop1] == 0  )                     //此处表示如果begin节点到loop1节点不能直达的话,就直接把经过node节点的距离当做最短距离 
  66.                 {   dist[loop1] = dist[node] + mg.edges[node][loop1];  path[loop1] = node;} 
  67.                 else if((dist[node] + mg.edges[node][loop1] ) < dist[loop1])        //否则的话,就进行判断如果进过node节点的路径长度小于直达的路径长度,进把距离的值重新赋值 
  68.                 {   dist[loop1] = dist[node] + mg.edges[node][loop1];  path[loop1] = node;} 
  69.             } 
  70.         } 
  71.         ++count; 
  72.     } 
  73.  
  74.  
  75.     for(loop1 = 0; loop1 < mg.n; ++loop1) 
  76.     { 
  77.         cout<<"shortest path to "<<loop1<<" is "<<dist[loop1]<<endl; 
  78.     } 
  79.     DispPath(path,begin,6); 
  80.  
  81. /* 
  82.     采用弗洛伊德算法求每对顶点之间的最短路径 
  83.     A[i][j]数组用以保存从i节点到j节点之间的最短路径,初始的值为顶点之间的权值 
  84.     path[i][j]数组用以保存上一个节点的编号,即中间节点的编号,初始化的值为-1,表示没有中间节点 
  85.  
  86.     其步骤是: 
  87.     1. 确定顶点i到j之间是否有通过0节点的路径,如果有的话,此时表示A[i][0] != 0  A[0][j] != 0 成立,然后判断这两段的长度是否比直达的长度小,如果小的话,就进行最短路径的替换(A数组保存的最短的路径),同时把path数组相应的节点置为0,表示从节点i到节点j之间存在一个中间节点为0 
  88.     2. 考察顶点i到j之间是否有通过1节点的路径,如果有的话,即按照上面的步骤进行,否则直接跳过 
  89.     3. 重复上述步骤2 
  90.  
  91.   通过比较知道:狄克斯特拉算法是求某个特定的顶点到其余顶点之间的最短路径    此算法表现的是特定的顶点与其余顶点之间的关系  
  92.                 而弗洛伊德算法则是求每对顶点之间的最短路径                   此算法表现的是定点与顶点的关系 
  93. */ 
  94.  
  95. void Floyd(MGraph mg) 
  96.     int path[MAXSIZE][MAXSIZE], A[MAXSIZE][MAXSIZE];        //path[i][j]数组保存的是从顶点i到顶点j中间节点编号不大于k的路径的上一个节点的编号 
  97.     int ivar = mg.n, loop1 = 0, loop2 = 0, loop3 = 0;                                                             //A[i][j]数组保存的是从i到j的最短距离 
  98.      
  99.     for(loop1 = 0; loop1 < ivar; ++loop1) 
  100.         for(loop2 = 0; loop2 < ivar; ++loop2) 
  101.         { 
  102.             path[loop1][loop2] = -1; 
  103.             A[loop1][loop2]    = mg.edges[loop1][loop2];                          //对A数组进行初始化为边的权值,0表示两个顶点之间不可以直达或没路径 
  104.         } 
  105.  
  106.     for(loop1 = 0; loop1 < ivar; ++loop1) 
  107.     {   for(loop2 = 0; loop2 < ivar; ++loop2) 
  108.             for(loop3 = 0; loop3 < ivar; ++loop3) 
  109.             {//此处应该以A数组为对象,不应该以mg.edges数组为对象。 
  110.             //  if(loop2 != loop3 && mg.edges[loop2][loop1] != 0 && mg.edges[loop1][loop3] != 0 && (A[loop2][loop3] == 0 || (mg.edges[loop2][loop1] + mg.edges[loop1][loop3]) < A[loop2][loop3])) 
  111.                 if(loop2 != loop3 && A[loop2][loop1] != 0 && A[loop1][loop3] != 0 && (A[loop2][loop3] == 0 || (A[loop2][loop1] + A[loop1][loop3]) < A[loop2][loop3])) 
  112.                 { 
  113.                     A[loop2][loop3]    = A[loop2][loop1] + A[loop1][loop3]; 
  114.                     path[loop2][loop3] = loop1;  
  115.                 } 
  116.             } 
  117.         for(loop2 = 0; loop2 < ivar; ++loop2) 
  118.         {   for(loop3 = 0; loop3 < ivar; ++loop3) 
  119.                 cout<<A[loop2][loop3]<<' ';          
  120.             cout<<endl; 
  121.         } 
  122.         cout<<endl;  
  123.     } 
  124.  
  125.  
  126. void MinTree(MGraph mg,int begin) 
  127.     int dist[MAXSIZE], path[MAXSIZE], v[MAXSIZE]; 
  128.     int loop1 = 0, count = 1,node = 0, min = 10000, pre = 0,ivar = mg.n; 
  129.  
  130.     for(loop1 = 0; loop1 < ivar; ++loop1) 
  131.     { 
  132.         v[loop1]    = 0;         
  133.         dist[loop1] = mg.edges[begin][loop1]; 
  134.         if(mg.edges[begin][loop1] != 0) 
  135.         { path[loop1] = begin; } 
  136.         else 
  137.         { path[loop1] = -1; } 
  138.     } 
  139.      
  140.     v[begin] = 1; 
  141.     count    = 1; 
  142.     while(count < ivar) 
  143.     { 
  144.         min = 10000; 
  145.         for(loop1 = 0; loop1 < ivar; ++loop1) 
  146.         { 
  147.             if(v[loop1] == 0 && dist[loop1] != 0 && min > dist[loop1]) 
  148.             { 
  149.                 node = loop1; 
  150.                 min  = dist[loop1]; 
  151.             } 
  152.         } 
  153.         v[node] = 1;   
  154.         for(loop1 = 0; loop1 < ivar; ++loop1) 
  155.         {            
  156.             if(mg.edges[node][loop1] != 0 && (dist[loop1] == 0 || (dist[node] + mg.edges[node][loop1]) < dist[loop1] )) 
  157.             { 
  158.                 dist[loop1] = mg.edges[node][loop1]  + dist[node]; 
  159.                 path[loop1] = node; 
  160.             } 
  161.         } 
  162.         ++count; 
  163.     } 
  164.  
  165.     for(loop1 = 0; loop1 < ivar; ++loop1) 
  166.     { 
  167.         if(v[loop1] == 1 && loop1 != begin) 
  168.         { 
  169.             pre = path[loop1]; 
  170.             cout<<"line is "<<pre<<" to "<<loop1<<" weight is "<<mg.edges[pre][loop1]<<endl; 
  171.         } 
  172.     } 
  173.  
  174.  
  175. void Centerp(MGraph mg) 
  176.     int loop1 = 0, loop2 = 0, loop3 = 0, max = -1, maxindex = 0, ivar = mg.n; 
  177.     int A[MAXSIZE][MAXSIZE], path[MAXSIZE][MAXSIZE]; 
  178.  
  179.     for(loop1 = 0; loop1 < ivar; ++loop1) 
  180.         for(loop2 = 0; loop2 < ivar; ++loop2) 
  181.         { 
  182.             A[loop1][loop2]    = mg.edges[loop1][loop2]; 
  183.             path[loop1][loop2] = -1; 
  184.         } 
  185.     for(loop1 = 0; loop1 < ivar; ++loop1) 
  186.         for(loop2 = 0; loop2 < ivar; ++loop2) 
  187.             for(loop3 = 0; loop3 < ivar; ++loop3) 
  188.             { 
  189.                 if(A[loop2][loop1] != 0 && A[loop1][loop3] != 0 && (A[loop2][loop3] == 0 || (A[loop2][loop1] + A[loop1][loop3]) < A[loop2][loop3]) ) 
  190.                 { 
  191.                     A[loop2][loop3]    = A[loop2][loop1] + A[loop1][loop2]; 
  192.                     path[loop2][loop3] = loop1; 
  193.                 } 
  194.             } 
  195.     for(loop1 = 0; loop1 < ivar; ++loop1) 
  196.     { 
  197.         for(loop2 = 0; loop2 < ivar; ++loop2) 
  198.         { 
  199.             if(max < A[loop2][loop1]) 
  200.             { 
  201.                 max      = A[loop2][loop1]; 
  202.                 maxindex = loop1; 
  203.             } 
  204.         } 
  205.     } 
  206.     cout<<"the center node is : "<<maxindex<<endl; 

 

你可能感兴趣的:(算法,最短路径,图)