1393:联络员(liaison)
时间限制: 1000 ms 内存限制: 65536 KB
提交数:6959 通过数: 3538
【题目描述】
Tyvj已经一岁了,网站也由最初的几个用户增加到了上万个用户,随着Tyvj网站的逐步壮大,管理员的数目也越来越多,现在你身为Tyvj管理层的联络员,希望你找到一些通信渠道,使得管理员两两都可以联络(直接或者是间接都可以)。Tyvj是一个公益性的网站,没有过多的利润,所以你要尽可能的使费用少才可以。
目前你已经知道,Tyvj的通信渠道分为两大类,一类是必选通信渠道,无论价格多少,你都需要把所有的都选择上;还有一类是选择性的通信渠道,你可以从中挑选一些作为最终管理员联络的通信渠道。数据保证给出的通行渠道可以让所有的管理员联通。
【输入】
第一行n,m表示Tyvj一共有n个管理员,有m个通信渠道;
第二行到m+1行,每行四个非负整数,p,u,v,w 当p=1时,表示这个通信渠道为必选通信渠道;当p=2时,表示这个通信渠道为选择性通信渠道;u,v,w表示本条信息描述的是u,v管理员之间的通信渠道,u可以收到v的信息,v也可以收到u的信息,w表示费用。
【输出】
最小的通信费用。
【输入样例】
5 6
1 1 2 1
1 2 3 1
1 3 4 1
1 4 1 1
2 2 5 10
2 2 5 5
【输出样例】
9
【提示】
【样例解释】
1-2-3-4-1存在四个必选渠道,形成一个环,互相可以到达。需要让所有管理员联通,需要联通2号和5号管理员,选择费用为5的渠道,所以总的费用为9。
【注意】
U,v之间可能存在多条通信渠道,你的程序应该累加所有u,v之间的必选通行渠道
【数据范围】
对于30%的数据,n≤10,m≤100;
对于50%的数据, n≤200,m≤1000
对于100%的数据,n≤2000,m≤10000
并查集 连通性:先把标记为1的结点的w直接进行求和,利用并查集合并圈子,
把标记为2的结点,存起来,按照结构体排序(规则:w的从小到大),然后合并到第1个圈子,
如果uv不同父亲的话,则合并,并求和w。如果在一个圈子就不合并不求和
输出sum
*/
#include<bits/stdc++.h>
using namespace std;
typedef struct node
{
int u,v,w;
bool operator<(const node nd1)const
{
return w<nd1.w;
}
}node ;
int n,m;
node nd[10005];
int father[2005];
void init()
{
int i;
for(i=1;i<=n;i++)
{
father[i]=i;
}
}
int Find(int x)
{
if(x!=father[x])
father[x]=Find(father[x]);
return father[x];
}
int main()
{
cin>>n>>m;
int i;
init();// 并查集初始化
int sum=0;//花费之和
int cnt=0;// 存起来这样的结点
for(i=1;i<=m;i++)
{
int flag,u,v,w;
cin>>flag>>u>>v>>w;
if(flag==1)// 必须花费
{
sum=sum+w;
// 合并圈子
int uu=Find(u);
int vv=Find(v);
if(uu!=vv)
father[vv]=uu;
}
else
{
nd[++cnt]={u,v,w};// 标记为2的结点存起来
}
}
sort(nd+1,nd+cnt+1);// 对标记为2的结点排序,寻找连通
for(i=1;i<=cnt;i++)
{
int uu=Find(nd[i].u);
int vv=Find(nd[i].v);
if(uu!=vv)
{
sum=sum+nd[i].w;
father[vv]=uu;
}
}
cout<<sum<<endl;
return 0;
}