洛谷 P2820 局域网(prim)

题目链接

https://www.luogu.org/problemnew/show/P2820

题目背景

某个局域网内有n(n<=100)台计算机,由于搭建局域网时工作人员的疏忽,现在局域网内的连接形成了回路,我们知道如果局域网形成回路那么数据将不停的在回路内传输,造成网络卡的现象。因为连接计算机的网线本身不同,所以有一些连线不是很畅通,我们用f(i,j)表示i,j之间连接的畅通程度,f(i,j)值越小表示i,j之间连接越通畅,f(i,j)为0表示i,j之间无网线连接。

题目描述

需要解决回路问题,我们将除去一些连线,使得网络中没有回路,并且被除去网线的Σf(i,j)最大,请求出这个最大值。

输入输出格式

输入格式:

第一行两个正整数n k

接下来的k行每行三个正整数i j m表示i,j两台计算机之间有网线联通,通畅程度为m。

输出格式:

一个正整数,Σf(i,j)的最大值

输入输出样例

输入样例:

5 5
1 2 8
1 3 1
1 5 3
2 4 5
3 4 2

输出样例:

8

说明

f(i,j)<=1000

解题思路

首先,我们需要了解一下有关最小生成树的知识。

名词解释:

  • 最小生成树:对于带权图,权值和最小的生成树

  • 最小瓶颈生成树:对于带权图,最大权值最小的生成树

  • 最小生成树一定是最小瓶颈生成树

蒟蒻不理解,换一种解释:

  • 生成树:在一幅图中将所有n个点连接起来的n-1条边所形成的树

  • 最小生成树:边权之和最小的生成树

算法:

  Prim算法或者Kruskal算法。但是一般都会使用Kruskal算法,因为它实用性更强,更加灵活,而且不用建图。

  但是在这里我们介绍prim算法

prim算法:

  prim算法的思想和dijstra很像——都是红白点思想。初始状态下,除第一个点(可以是任意一个点)为红点外,所有的点都是白点,每一次循环找出距离这个红点最近的白点,把它变成红点,然后再用这个点去更新其他的白点。就像这样,直到所有的点都变成红点。

这个题就是求被除去的长度最大,就是剩下的总长度最小,剩下的总长度就是最小生成树的边权和,被除去的总长度=原来总长度-最小生成树的边权和。

其中对memset不懂的推荐我的另一篇博客:https://www.cnblogs.com/yinyuqin/p/10162716.html

具体实现请见代码:

 1 #include<iostream>
 2 #include<cstring>        //memset的头文件 
 3 using namespace std;
 4 int m[105][105],n,k,cnt1,cnt2;    //m用来存图(邻接矩阵存图) 
 5 int a[105];                //a[i]存的是每个点到最小生成树的最短距离 
 6 int ok[105];            //ok[i]存的是i这个点是否已经在最小生成树上 
 7 int main(){
 8     cin>>n>>k;
 9     for(int i=1;i<=k;i++){
10         int a,b,c;
11         cin>>a>>b>>c;
12         m[a][b]=m[b][a]=c;        //这个一个无向图。 
13         cnt1+=c;                //存原图的边权和 
14     }
15     memset(a,0x7f,sizeof(a));    //先把每个点到最小生成树的距离设置为无穷大。 
16     a[1]=0;                        //将点1加入最小生成树,点1到最小生成树的距离设置为0。 
17     for(int j=1;j<=n;j++){        //要保证每一个点都在最小生成树里面,就需要循环n次。 
18         int minn=0;
19         for(int i=1;i<=n;i++){    //找到距离最小生成树最小的白点 
20             if(!ok[i]&&((a[i]<a[minn])||minn==0)) minn=i;
21         }
22         ok[minn]=1;                //将白点变为红点,加入最小生成树 
23         for(int i=0;i<=n;i++){    //更新与这个点相邻的其他白点距最小生成树的距离 
24             if(m[minn][i]&&!ok[i]) a[i]=min(m[minn][i],a[i]);//取最小值 
25         }
26     }
27     for(int i=1;i<=n;i++) cnt2+=a[i];    //统计最小生成树的边权和 
28     cout<<cnt1-cnt2;            //相减即为答案 
29     return 0;
30 }
AC代码

 

posted @ 2019-04-27 17:12  尹昱钦  阅读(393)  评论(0编辑  收藏  举报