题意:给定n个点,你的任务是让它们都连通。你可以新建一些边,费用等于两点距离的平方(当然越小越好),另外还有几种“套餐”,可以购买,你购买的话,那么有些边就可以连接起来,

每个“套餐”,也是要花费的,让你求出最少花费。

析:首先想到的是把所有情况都考虑算一下,然后找出最少的,先算没有“套餐”的,然后算有的,用二进制枚举的话,总时间复杂度为O(2qn2+n2logn),这个时间复杂度太大了吧,肯定会超时,

那么我们就可以优化一下,首先先算出来最小生成树,并且把每条边都保存下来,那么加了“套餐”之后,就不用全部枚举了,这是一个优化,然后在买“套餐”后,那么有的权值就变成了0,

这个也不要加上,再重新枚举,我们在之前就把它们连接上就OK。其他的和最小生成树一样。

代码如下:

#include <cstdio>
#include <iostream>
#include <algorithm> using namespace std;
const int maxn = 1000 + 10;
struct node{
int u, v, w;
bool operator < (const node &p) const {
return w < p.w;
}
};
node a[maxn*maxn/2];
int p[maxn], x[maxn], y[maxn], n, m, q[8][maxn], c[2][10], indx, l[maxn]; int Find(int x){
return x == p[x] ? x : p[x] = Find(p[x]);
} int Kruskal(){//最小生成树
int ans = 0;
l[0] = 0;
for(int i = 0; i < indx; ++i){
int x = Find(a[i].u);
int y = Find(a[i].v);
if(x != y){
ans += a[i].w;
p[x] = y;
l[++l[0]] = i;//把最小生成树的边都保存下来
}
}
return ans;
} int Kruskal2(){//买“套餐”后的最小生成树
int ans = 0;
for(int i = 1; i <= l[0]; ++i){
int ii = l[i];
int x = Find(a[ii].u);
int y = Find(a[ii].v);
if(x != y){
p[x] = y;
ans += a[ii].w;
}
}
return ans;
} int main(){
// freopen("int.txt", "r", stdin);
int T; cin >> T;
while(T--){
scanf("%d %d", &n, &m);
for(int i = 0; i < m; ++i){
scanf("%d %d", &c[0][i], &c[1][i]);
for(int j = 0; j < c[0][i]; ++j)
scanf("%d", &q[i][j]);
}
for(int i = 0; i < n; ++i)
scanf("%d %d", &x[i], &y[i]); indx = 0;
for(int i = 0; i < n; ++i)//计算权植
for(int j = i+1; j < n; ++j){
a[indx].u = i+1;
a[indx].v = j+1;
a[indx++].w = (x[i]-x[j])*(x[i]-x[j]) + (y[i]-y[j])*(y[i]-y[j]);
} sort(a, a+indx);
for(int i = 0; i <= n; ++i) p[i] = i;
int ans = Kruskal();//计算不买”套餐“时的最小费用 for(int i = 0; i < (1<<m); ++i){//二进制法枚举
int cost = 0;
for(int j = 0; j <= n; ++j) p[j] = j; for(int j = 0; j < m; ++j){
if(i&(1<<j)){
cost += c[1][j];
for(int k = 0; k < c[0][j]-1; ++k){
int xx = Find(q[j][k]);
int yy = Find(q[j][k+1]);
if(xx != yy) p[xx] = yy;
}
}
}
ans = min(ans, cost + Kruskal2());//加上买套餐的费用,更新最小值
} printf("%d\n", ans);
if(T) printf("\n");
}
return 0;
}

UVa 1151 Buy or Build (最小生成树+二进制法暴力求解)的更多相关文章

  1. UVa 1151 Buy or Build【最小生成树】

    题意:给出n个点的坐标,现在需要让这n个点连通,可以直接在点与点之间连边,花费为两点之间欧几里得距离的平方,也可以选购套餐,套餐中所含的点是相互连通的 问最少的花费 首先想kruskal算法中,被加入 ...

  2. UVA 1151 Buy or Build MST(最小生成树)

    题意: 在平面上有n个点,要让所有n个点都连通,所以你要构造一些边来连通他们,连通的费用等于两个端点的欧几里得距离的平方.另外还有q个套餐,可以购买,如果你购买了第i个套餐,该套餐中的所有结点将变得相 ...

  3. UVa 1151 - Buy or Build(最小生成树)

    链接: https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...

  4. UVA 1151 Buy or Build (最小生成树)

    先求出原图的最小生成树,然后枚举买哪些套餐,把一个套餐内的点相互之间边权为0,直接用并查集缩点.正确性是基于一个贪心, 在做Kruskal算法是,对于没有进入最小生成树的边,排序在它前面的边不会减少. ...

  5. UVA - 1151 Buy or Build (买还是建)(并查集+二进制枚举子集)

    题意:平面上有n个点(1<=n<=1000),你的任务是让所有n个点连通.可以新建边,费用等于两端点欧几里德距离的平方.也可以购买套餐(套餐中的点全部连通).问最小费用. 分析: 1.先将 ...

  6. UVA 1151 Buy or Build (MST最小生成树,kruscal,变形)

    题意: 要使n个点之间能够互通,要使两点直接互通需要耗费它们之间的欧几里得距离的平方大小的花费,这说明每两个点都可以使其互通.接着有q个套餐可以选,一旦选了这些套餐,他们所包含的点自动就连起来了,所需 ...

  7. uva 1151 - Buy or Build poj 2784 Buy or Build(最小生成树)

    最小生成树算法简单 只是增加了一些新的东西,对于需要最小生成树算法 和中 并检查使用的一系列 还有一些更深入的了解. 方法的一些复杂问题 #include<cstdio> #include ...

  8. UVA 1151 买还是建(最小生成树)

    买还是建 紫书P358 [题目链接]买还是建 [题目类型]最小生成树 &题解: 这题真的心累,看了3天,最后照着码还是wa,先放lrj代码,以后再看吧 &代码: // UVa1151 ...

  9. uva 1151最小生成树

    先求一次最小生成树,可以排除n*(n*1)/2-(n-1)条边,每次利用二进制法枚举套餐的选择,套餐中的点直接处理,如果两个套餐有公共点直接合并,他们一定连通,然后枚举第一步最小生成树得到的n-1条边 ...

随机推荐

  1. jquery类的创建方式及关键字new的原理

    一.由jQuery创建类引发的问题 在用jQuery选择器时候,可以通过下面两种方式获取元素,并得到一个jQuery对象. var d1 = jQuery('#demo01'); var d2 = n ...

  2. 关于AM335X移植SDIO WIFI的简易教程(转)

    最近应一个朋友邀请,帮他移植了SDIO WIFI到3.2版本内核.因为之前已经成功移植了3.14内核,所以整个过程花了一个下午就完成了.话不多说,先交待一下平台: CPU:TI AM3352 600M ...

  3. Tomcat服务器学习和使用(二)

    一.打包JavaWeb应用 在Java中,使用"jar"命令来对将JavaWeb应用打包成一个War包,jar命令的用法如下:

  4. Hadoop编译安装

    本文转载自:http://www.cnblogs.com/shishanyuan/p/4701646.html 1.编译Hadooop 1.1 搭建环境 1.1.1 安装并设置maven 1. 下载m ...

  5. appium启动APP时避免重新安装的问题

    from appium import webdriverfrom time import sleepimport os #获取apk的绝对路径desired_cups = {}#设备平台desired ...

  6. Java 遍历文件夹里面的全部文件、指定文件

    Java 手册 listFiles public File[] listFiles(FileFilter filter) 返回抽象路径名数组,这些路径名表示此抽象路径名表示的目录中满足指定过滤器的文件 ...

  7. CentOS6.5 安装mysql-5.7.9

    转自:http://forrest-lv.iteye.com/blog/2260703 安装前,需要检查是否已经有mysql服务进程,是否已经装过mysql;  这点很重要,我之前安装CentOS的同 ...

  8. HTTP协议响应头之Transfer-Encoding:分块传输详解

    Http Connection有两种连接方式:短连接和长连接:短连接即一次请求对应一次TCP连接的建立和销毁过程,而长连接是多个请求共用同一个连接这样可以节省大量连接建立时间提高通信效率.目前主流浏览 ...

  9. 【POJ】1185 炮兵阵地(状压dp)

    题目 传送门:QWQ 分析 看到$ M<=10 $考虑状压. 然后把每行都压一下,那么每个状态相关的就是上一行和上上行的状态. 然后枚举. 然后复杂度最坏是$ O(100 \times 1024 ...

  10. 年薪30万的Android程序员必须知道的帖子

    https://github.com/Trinea/android-open-project