题意:

  在平面上有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. FluentAPI深入

    1.  HasMaxLenght 设定字段得最大长度: static void Main(string[] args) { using (TestDbContext ctx = new TestDbC ...

  2. 网易2017校园招聘算法题c语言实现源码

    题目: 给定一个数组,除了一个数出现1次之外,其余数都出现3次.找出出现一次的数.如:{1, 2, 1, 2, 1, 2, 7}, 找出7. 格式: 第一行输入一个数n,代表数组的长度,接下来一行输入 ...

  3. Extjs msgTarget 提示位置

    extjs msgTarget 有效值包括: qtip:显示一个浮动的提示消息 title:显示一个浏览器浮动提示消息 under:在字段下面显示一个提示消息,使用under时要注意表单的高度 sid ...

  4. 【DNS】DNS的几个基本概念

    一. 根域 就是所谓的“.”,其实我们的网址www.baidu.com在配置当中应该是www.baidu.com.(最后有一点),一般我们在浏览器里输入时会省略后面的点,而这也已经成为了习惯. 根域服 ...

  5. 网站前后台分离 图片 flash 视频 等文件的共享问题

    在网上找了,没有说到点子上的,不详细 问了有经验的同事,要建立 文件服务器,就是一个IIS 下的新网站,网站是共享图片 文件使用的专用网站 后台上传的图片保存在 文件服务器即 文件共享专用的网站目录地 ...

  6. mysql外网链接

    1:设置mysql的配置文件     /etc/mysql/my.cnf     找到 bind-address  =127.0.0.1  将其注释掉://作用是使得不再只允许本地访问:  重启mys ...

  7. Educational Codeforces Round 42 (Rated for Div. 2) A

    A. Equator time limit per test 2 seconds memory limit per test 256 megabytes input standard input ou ...

  8. Java面试题之Oracle 支持哪三种事务隔离级别

    Oracle 支持三种事务隔离级别: 1.读已提交:(默认) 2.串行化: 3.只读模式

  9. 托福、雅思和GRE的区别

    托福雅思GRE区别在哪里?对于准备申请美国硕士生的同学们来说,必须了解这一点,才能根据自身实际情况进行有针对性的复习,下面我们来进行详细介绍,为同学们指点迷津. - GRE是由美国教育考试服务处(Ed ...

  10. Mysql事务隔离级

    转自:http://xm-king.iteye.com/blog/770721 SQL标准定义了4类隔离级别,包括了一些具体规则,用来限定事务内外的哪些改变是可见的,哪些是不可见的.低级别的隔离级一般 ...