UVA 1151 Buy or Build MST(最小生成树)
题意:
在平面上有n个点,要让所有n个点都连通,所以你要构造一些边来连通他们,连通的费用等于两个端点的欧几里得距离的平方。另外还有q个套餐,可以购买,如果你购买了第i个套餐,该套餐中的所有结点将变得相互连通,第i个套餐的花费为ci。求最小花费。
思路:
在这里我们可以采取枚举所有可能 + K算法来得出答案,比如这里有三个套餐,我们利用二进制枚举 001、010、011 、100、 101、 110、 111 分别代表第一个和第二个不要,要第三个(001);不要第一个和第三个,要第二个(010).......即 0 代表不要, 1 代表要,然后把要的套餐中的所有点都连通,再用K算法求剩下的未连接的点的最小生成树。
注意:
在套餐中合并点时不能单纯地让pre[i] = pre[1](i > 1);pre[i]数组代表 pre[i] 和 i 在一个集合里面(并查集);举个栗子:
有一个套餐是:
4 10 2 3 4 5
含义是购买这个套餐中可以让四个点连通,分别是2,3,4,5号点,费用为 10;如果让 pre[3] = pre[4]=pre[5] = 2;
那么假设还有个套餐:
3 9 1 5 3
含义如上 ,如果再写pre[5] = pre[3] = 1;那么假设我购买了这俩个套餐,本应该2 3 4 5 1都在一个集合里面的,但是按照上面那么写 则 2 4 是一个集合, 1 3 5 是一个集合。不符合我的意思,所以购买套餐合并里面的点时应该写成pre[i] = Find(pre[1]);前提是这俩个不在一个集合里面。
代码:
#include <bits/stdc++.h>
#define prln(x) cout<<(x)<<endl
using namespace std;
typedef long long LL; const double PI = acos(-1);
const double ESP = 1e-8;
const int MAXN = 1000 + 3;
const int MOD = 1e9 + 7;
int pre[MAXN]; typedef struct Point{ //题目中给的点
int x;
int y;
}Po; typedef struct Buy{ //套餐
int m; //购买该套餐可以合并点的个数
int ci; //购买该套餐的费用
int a[MAXN]; //这个套餐可以合并的点的编号
int flag; //是否要购买这个套餐,对每个套餐的这个值进行二进制枚举
}Bu; typedef struct City{ //用来存储图
int u;
int v;
int w;
}Ci; Ci edge[MAXN * MAXN / 2 + 3];
Po pt[MAXN];
Bu buy[11]; int Find(int x) //并查集
{
return x == pre[x] ? x : pre[x] = Find(pre[x]);
} void Stpre(int n)
{
for(int i = 0; i <= n; i++)
pre[i] = i;
} void Ststu()
{
memset(&pt,0,sizeof(Po));
memset(&buy,0,sizeof(Bu));
memset(&edge,0,sizeof(Ci));
} int Ojld(Point a, Point b)
{
int xx = a.x - b.x;
int yy = a.y - b.y;
return xx * xx + yy *yy;
} int mycmp(City a, City b)
{
return a.w < b.w;
} int ksu(int l)//K算法
{
int ans= 0;
for(int i = 1; i<= l; i++)
{
int fv = Find(edge[i].v);
int fu = Find(edge[i].u);
if(fu != fv)
{
pre[fu] = pre[fv];
ans += edge[i].w;
}
}
return ans;
} int main()
{
//freopen("input.txt","r",stdin);
int t;
cin >> t;
while(t--)
{
Ststu();
int n, q;
scanf("%d%d",&n, &q);
for(int i = 1; i <= q; i++)
{
scanf("%d",&buy[i].m);
scanf("%d",&buy[i].ci);
for(int j = 1; j <= buy[i].m; j++)
scanf("%d",&buy[i].a[j]);
}
for(int i = 1; i <= n; i++)
scanf("%d%d",&pt[i].x, &pt[i].y);
int sum = 0;
for(int i = 1; i < n; i++)
{
for(int j = i + 1; j <= n; j++)
{
sum++;
edge[sum].u = i;
edge[sum].v = j;
edge[sum].w = Ojld(pt[i], pt[j]);
//printf("%d %d %d %d\n",sum,i,j,edge[sum].w);
}
}
sort(edge + 1, edge + sum + 1 , mycmp); int ans = 0x7F7F7F7F;
for(int i = 0; i < (1 << q); i++) //二进制枚举
{
Stpre(n);
int temp = i;
int mst = 0;
for(int j = 1; j <= q; j++)
{
if(temp & 1)
{
mst += buy[j].ci;
for(int k = 2; k <= buy[j].m; k++)
{
int fx = Find ( buy[j].a[1] );
int fy = Find( buy[j].a[k] );
if(fy != fx)
pre[fy] = pre[fx];
}
}
temp >>= 1;
}
mst += ksu(sum);
ans = min(ans, mst);
}
printf("%d\n",ans);
if(t)prln("");
}
return 0;
}
UVA 1151 Buy or Build MST(最小生成树)的更多相关文章
- UVa 1151 - Buy or Build(最小生成树)
链接: https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...
- UVA 1151 Buy or Build (最小生成树)
先求出原图的最小生成树,然后枚举买哪些套餐,把一个套餐内的点相互之间边权为0,直接用并查集缩点.正确性是基于一个贪心, 在做Kruskal算法是,对于没有进入最小生成树的边,排序在它前面的边不会减少. ...
- UVa 1151 Buy or Build【最小生成树】
题意:给出n个点的坐标,现在需要让这n个点连通,可以直接在点与点之间连边,花费为两点之间欧几里得距离的平方,也可以选购套餐,套餐中所含的点是相互连通的 问最少的花费 首先想kruskal算法中,被加入 ...
- UVa 1151 Buy or Build (最小生成树+二进制法暴力求解)
题意:给定n个点,你的任务是让它们都连通.你可以新建一些边,费用等于两点距离的平方(当然越小越好),另外还有几种“套餐”,可以购买,你购买的话,那么有些边就可以连接起来, 每个“套餐”,也是要花费的, ...
- UVA 1151 Buy or Build (MST最小生成树,kruscal,变形)
题意: 要使n个点之间能够互通,要使两点直接互通需要耗费它们之间的欧几里得距离的平方大小的花费,这说明每两个点都可以使其互通.接着有q个套餐可以选,一旦选了这些套餐,他们所包含的点自动就连起来了,所需 ...
- uva 1151 - Buy or Build poj 2784 Buy or Build(最小生成树)
最小生成树算法简单 只是增加了一些新的东西,对于需要最小生成树算法 和中 并检查使用的一系列 还有一些更深入的了解. 方法的一些复杂问题 #include<cstdio> #include ...
- UVA - 1151 Buy or Build (买还是建)(并查集+二进制枚举子集)
题意:平面上有n个点(1<=n<=1000),你的任务是让所有n个点连通.可以新建边,费用等于两端点欧几里德距离的平方.也可以购买套餐(套餐中的点全部连通).问最小费用. 分析: 1.先将 ...
- 紫书 例题 11-3 UVa 1151 (有边集的最小生成树+二进制枚举子集)
标题指的边集是说这道题的套餐, 是由几条边构成的. 思路是先做一遍最小生成树排除边, 因为如果第一次做没有加入的边, 到后来新加入了很多权值为0的边,这些边肯定排在最前面,然后这条边的前面的那些边肯定 ...
- UVA 1151 买还是建(最小生成树)
买还是建 紫书P358 [题目链接]买还是建 [题目类型]最小生成树 &题解: 这题真的心累,看了3天,最后照着码还是wa,先放lrj代码,以后再看吧 &代码: // UVa1151 ...
随机推荐
- Python 3基础教程10-全局变量和局部变量
本文来讲讲全局变量和局部变量,前面学习了函数的基本使用,所以,这里就要注意变量的使用和访问权限. 试试下面的demo.py
- Java基础-4变量与数据类型
变量:变量是Java程序中的一个基本存储单元.变量是一个标识符.类型及一个可选初始值的组合定义.所有的变量都有一个作用域,即变量在某一区域有效. 基本的变量声明方式如下: int a; float b ...
- Opencv3.2.0安装包
这个资源是Opencv3.2.0安装包,包括Windows软件包,Android软件包,IOS软件包,还有opencv的源代码:需要的下载吧. 点击下载
- React01
目录 React-day01 入门知识 React介绍 官网 React开发环境初始化 SPA 脚手架初始化项目(方便,稳定)* 通过webpack进行初始化 配置镜像地址 开发工具配置 元素渲染 组 ...
- Codeforces Round #327 (Div2) A~E
CodeForces 591A 题意:在距离为L的两端A,B,相向发射魔法,a(以P1的速度)-->B,A<--b(以P2的速度).假设a-->B,途中相遇,则返回到原点A<- ...
- String类型的方法使用
String.equals()方法源代码: public boolean equals(Object anObject) { if (this == anObject) { return true; ...
- js中prop和attr区别
首先 attr 是从页面搜索获得元素值,所以页面必须明确定义元素才能获取值,相对来说比较慢. 如: <input name='test' type='checkbox'> $('input ...
- 7月11日day3总结
今天学习过程和总结 一 1.输出流的字符流.字节流 2.加锁.多线程的理解,产生的原因.cpu同时运行最大数.其他的都在及时切换.1.继承Thred类,重写run方法. 2.实现Runnable接口. ...
- Nodejs express框架 浅析
http://www.expressjs.com.cn/ 1. 中间件 ①挂载中间件的函数:app.use var http = require('http'); var express = requ ...
- java 竖线分割字符串的问题
java 竖线分割字符串的问题 例1: String[] paraStr = "6010;320100;A".split(";"); System.out.pr ...