题意:

  在平面上有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(最小生成树)的更多相关文章

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

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

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

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

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

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

  4. UVa 1151 Buy or Build (最小生成树+二进制法暴力求解)

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

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

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

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

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

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

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

  8. 紫书 例题 11-3 UVa 1151 (有边集的最小生成树+二进制枚举子集)

    标题指的边集是说这道题的套餐, 是由几条边构成的. 思路是先做一遍最小生成树排除边, 因为如果第一次做没有加入的边, 到后来新加入了很多权值为0的边,这些边肯定排在最前面,然后这条边的前面的那些边肯定 ...

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

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

随机推荐

  1. runloop和线程有什么关系?

    每条线程都有唯一的一个RunLoop对象与之对应的 主线程的RunLoop是自动创建并启动 子线程的RunLoop需要手动启动 子线程的RunLoop创建步骤如下: 获得RunLoop对象后要调用ru ...

  2. 打开Vim/Vi代码高亮

    由于新装Vim/Vi 默认是没有打开代码高亮配置的,就看到有朋友一次次到网上去找各种配置.其实Vim默认带来配置文件的样本的,只需拷贝过来就可使用. 在用户根目录(~)中新建vim的配置文件 .vim ...

  3. 《Cracking the Coding Interview》——第6章:智力题——题目6

    2014-03-20 01:14 题目:有100栈灯,一开始都关着.如果你按照n从1~100的顺序,每次都掰一下n的倍数的开关(开->关,关->开),那么到最后有多少灯是亮的? 解法:这个 ...

  4. windows下使用RoboCopy命令进行文件夹增量备份

    RoboCopy,它是一个命令行的目录复制命令,自从Windows NT 4.0 开始就成为windows 资源工具包的一部分,然后在Windows Vista.Windows 7和 Windows ...

  5. python代码简写(推导式 if else for in)

    c = a if a>b else b    //如果a>b返回a,否则返回b >>> a = 1 >>> b = 2 >>> c = ...

  6. 用js做的表单的增,删,以及全选

    <!DOCTYPE html><html> <head> <meta charset="UTF-8"> <title>创 ...

  7. 条件随机场(Conditional random field)

    条件随机场真是把我给折磨坏了啊,本以为一本小小的<统计学习方法>攻坚剩下最后一章,心情还是十分愉悦的,打算一口气把它看完,结果真正啃起来真是无比的艰难啊,每一句对我都好像是天书一般,怎么这 ...

  8. JavaScript里面的面向对象

    1.JavaScript里面没有类,但是利用函数可以起到类似的作用,例如简单的构造方法,跟Python差别不大 function f1(mame,age){ this.Name = name; thi ...

  9. django orm 的查询条件

    Django的ORM查询操作: 查询数据库操作是一个非常重要的技术.在Django中,查询一般就是使用filter.exclude.get三个方法来实现,在调用这些方法的时候传递不同的查询条件来实现复 ...

  10. windows 10 change default program bug

    windows 10 change default program bug https://www.isumsoft.com/windows-10/how-to-change-or-set-defau ...