vijos次小生成树

xiaomengxian的哥哥是一个游戏迷,他喜欢研究各种游戏。这天,xiaomengxian到他家玩,他便拿出了自己最近正在研究的一个游戏给xiaomengxian看。这个游戏是这样的:一个国家有N个城市,有些城市之间可以建设铁路,并且不同城市之间建设铁路的费用各不相同。问如何用最小的费用,使整个国家的各个城市之间能够互相到达。另外,铁路是双向的。xiaomengxian心想,这不是太简单了吗?这就是经典的MST问题。他的哥哥说,这个当然不算什么。关键是它还要求费用第二小的方案,这真是让人伤脑筋。xiaomengxian想了很久,也没有想出来,你能帮助他吗? 

费用第二小的方案的定义为:与费用最小的方案不完全相同,且费用值除费用最小的方案外最小。

同vijos 1070 网址:https://www.vijos.org/p/1070

题解:

 

次小生成树,就是求除了最小生成树之外最小的那个生成树。

下面介绍一下利用prim求次小生成树的主要步骤。

1.先求出来最小生成树。并将最小生成树任意两点之间路径当中的权值最大的那一条找出来,为什么要找最大的呢,因为生成树加入一条边之后一定构成了回路,那么肯定要去掉这个回路当中一条边才是生成树,那么,怎么去边才是次小的,那就去掉除了刚刚添加的一条边之外回路当中权值最大的一个,所以留下的就是最小的。

2.枚举最小生成树外的每一条边。找出最小的就是次小生成树。

代码如下,若还是不懂,代码中有注释,可自行参悟

 1 #include<cstdio>
 2 #include<algorithm>
 3 using namespace std; 
 4 int s,minn,g[501][501],m,x,y,z,ss,minn2,low[501],pre[501],low2[501][501]; 
 5 int i,mini,sum,n,t; bool b[501],used[501][501];
 6 int main() 
 7 {
 8      scanf("%d%d",&n,&m);
 9      for(i=1;i<=n;i++)         
10      {
11         g[i][i]=0;
12         for(int j=i+1;j<=500;j++)             
13             g[i][j]=g[j][i]=123456789;     
14     }
15     for(i=1;i<=m;i++)     
16     {         
17         scanf("%d%d%d",&x,&y,&z);     
18         g[y][x]=g[x][y]=z;//双向边
19     }     
20     for(i=1;i<=n;i++)         
21         low[i]=123456789,pre[i]=i;
22     pre[1]=0;  
23     low[1]=0;s=0;ss=923456789;
24     int k,kk;
25     while(1)     
26     {
27         minn=123456789;
28         for(i=1;i<=n;i++)         
29         {
30             if(!b[i]&&low[i]<minn)             
31             {                 
32                 minn=low[i];
33                 mini=i;
34             }
35         }
36         if(minn==123456789)break;
37         b[mini]=1;
38         s=s+minn;
39         used[mini][pre[mini]]=used[pre[mini]][mini]=1;
40         for(i=1;i<=n;i++)         
41         {
42             if(b[i]&&i!=mini)low2[i][mini]=low2[mini][i]=max(low2[i][pre[mini]],low[mini]);//记录最小生成树上的路径最大值
43             t=min(g[i][mini],g[mini][i]);
44             if(!b[i]&&t<low[i])
45                 low[i]=t,pre[i]=mini;//学过prim的都知道前半段是什么,后半句是记录路径
46         }
47     }
48     printf("Cost: %d\n",s);
49     for(int i=1;i<=n;i++)
50     {
51         for(int j=i+1;j<=n;j++)
52             if(g[i][j]!=123456789&&!used[i][j])
53                 ss=min(ss,s+g[i][j]-low2[i][j]);//枚举每一条边,看看替换该边是否有价值
54     }
55     if(ss==923456789)ss=-1;//如果没有边可以替换,即没有最小生成树,输出-1
56     printf("Cost: %d",ss);
57 }

 

posted @ 2018-08-08 11:33  蒟蒻--lichenxi  阅读(175)  评论(0编辑  收藏  举报