标题指的边集是说这道题的套餐, 是由几条边构成的。

思路是先做一遍最小生成树排除边, 因为如果第一次做没有加入的边, 到后来新加入了很多权值为0的边,这些边肯定排在最前面,然后这条边的前面的那些边肯定都要再扫一遍, 也就是这条边无论如何都不会选。

那么后来就是二进制枚举套餐, 从头开始, 加入套餐中的边然后权值加上套餐的权值, 然后把之前筛选下来的边做kruskal就ok了。

注意要对数据范围敏感, 这里套餐最多也就8个所以可以二进制枚举子集。

  1. #include<cstdio>
  2. #include<vector>
  3. #include<algorithm>
  4. #define REP(i, a, b) for(int i = (a); i < (b); i++)
  5. using namespace std;
  6. const int MAXN = 1123;
  7. int x[MAXN], y[MAXN], f[MAXN];
  8. int cost[MAXN], n, m, q, cnt, sum;
  9. struct node
  10. {
  11. int u, v, w;
  12. node(int u = 0, int v = 0, int w = 0) : u(u), v(v), w(w) {}
  13. bool operator < (const node& rhs) const
  14. {
  15. return w < rhs.w;
  16. }
  17. };
  18. vector<node> Edge, need;
  19. vector<int> k[9];
  20. int find(int x)
  21. {
  22. if(f[x] != x)
  23. f[x] = find(f[x]);
  24. return f[x];
  25. }
  26. int solve()
  27. {
  28. int ret = 0;
  29. REP(i, 0, need.size())
  30. {
  31. int u = find(need[i].u), v = find(need[i].v);
  32. if(u != v)
  33. {
  34. f[u] = v;
  35. ret += need[i].w;
  36. if(--cnt == 1) break;
  37. }
  38. }
  39. return ret;
  40. }
  41. int main()
  42. {
  43. int T;
  44. scanf("%d", &T);
  45. while(T--)
  46. {
  47. Edge.clear(); need.clear();
  48. scanf("%d%d", &n, &q);
  49. REP(i, 0, q)
  50. {
  51. int t, x; k[i].clear();
  52. scanf("%d%d", &t, &cost[i]);
  53. while(t--)
  54. {
  55. scanf("%d", &x);
  56. k[i].push_back(x);
  57. }
  58. }
  59. REP(i, 1, n + 1) scanf("%d%d", &x[i], &y[i]), f[i] = i;
  60. REP(i, 1, n + 1)
  61. REP(j, i + 1, n + 1)
  62. {
  63. int t = (x[i]-x[j]) * (x[i]-x[j]) + (y[i]-y[j]) * (y[i]-y[j]);
  64. Edge.push_back(node(i, j, t));
  65. }
  66. sort(Edge.begin(), Edge.end());
  67. int ans = 0; cnt = n;
  68. REP(i, 0, Edge.size())
  69. {
  70. int u = find(Edge[i].u), v = find(Edge[i].v);
  71. if(u != v)
  72. {
  73. need.push_back(Edge[i]);
  74. f[u] = v;
  75. ans += Edge[i].w;
  76. if(--cnt == 1) break; //注意是1
  77. }
  78. }
  79. REP(num, 0, 1 << q)
  80. {
  81. sum = 0; cnt = n;
  82. REP(i, 1, n + 1) f[i] = i;
  83. REP(i, 0, q)
  84. if(num & (1 << i))
  85. {
  86. sum += cost[i];
  87. REP(j, 0, k[i].size())
  88. {
  89. int u = find(k[i][j]), v = find(k[i][0]);
  90. if(u != v) f[u] = v, cnt--;
  91. }
  92. }
  93. sum += solve();
  94. ans = min(ans, sum);
  95. }
  96. printf("%d\n", ans);
  97. if(T) puts("");
  98. }
  99. return 0;
  100. }

紫书 例题 11-3 UVa 1151 (有边集的最小生成树+二进制枚举子集)的更多相关文章

  1. 紫书 例题 11-13 UVa 10735(混合图的欧拉回路)(最大流)

    这道题写了两个多小时-- 首先讲一下怎么建模 我们的目的是让所有点的出度等于入度 那么我们可以把点分为两部分, 一部分出度大于入度, 一部分入度大于出度 那么显然, 按照书里的思路,将边方向后,就相当 ...

  2. 紫书 例题8-3 UVa 1152(中途相遇法)

    这道题要逆向思维, 就是求出答案的一部分, 然后反过去去寻找答案存不存在. 其实很多其他题都用了这道题目的方法, 自己以前都没有发现, 这道题专门考这个方法.这个方法可以没有一直往下求, 可以省去很多 ...

  3. 紫书 例题8-12 UVa 12627 (找规律 + 递归)

    紫书上有很明显的笔误, 公式写错了.g(k, i)的那个公式应该加上c(k-1)而不是c(k).如果加上c(k-1)那就是这一次 所有的红气球的数目, 肯定大于最下面i行的红气球数 我用的是f的公式, ...

  4. 紫书 例题8-4 UVa 11134(问题分解 + 贪心)

     这道题目可以把问题分解, 因为x坐标和y坐标的答案之间没有联系, 所以可以单独求两个坐标的答案 我一开始想的是按照左区间从小到大, 相同的时候从右区间从小到大排序, 然后WA 去uDebug找了数据 ...

  5. 紫书 例题8-17 UVa 1609 (构造法)(详细注释)

    这道题用构造法, 就是自己依据题目想出一种可以得到解的方法, 没有什么规律可言, 只能根据题目本身来思考. 这道题的构造法比较复杂, 不知道刘汝佳是怎么想出来的, 我想的话肯定想不到. 具体思路紫书上 ...

  6. 紫书 例题 9-5 UVa 12563 ( 01背包变形)

    总的来说就是价值为1,时间因物品而变,同时注意要刚好取到的01背包 (1)时间方面.按照题意,每首歌的时间最多为t + w - 1,这里要注意. 同时记得最后要加入时间为678的一首歌曲 (2)这里因 ...

  7. 【uva 1151】Buy or Build(图论--最小生成树+二进制枚举状态)

    题意:平面上有N个点(1≤N≤1000),若要新建边,费用是2点的欧几里德距离的平方.另外还有Q个套餐,每个套餐里的点互相联通,总费用为Ci.问让所有N个点连通的最小费用.(2组数据的输出之间要求有换 ...

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

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

  9. 紫书 例题 10-26 UVa 11440(欧拉函数+数论)

    这里用到了一些数论知识 首先素因子都大于M等价与M! 互质 然后又因为当k与M!互质且k>M!时当且仅当k mod M! 与M!互质(欧几里得算法的原理) 又因为N>=M, 所以N!为M! ...

随机推荐

  1. node——将用户提交的数据写入data.json文件

    前续 当我们在进行将数据提交到某个网页时,需要将提交数据保存下来 1.提交数据 2.获得数据 3.保存数据 先看提交数据: <!DOCTYPE html> <html lang=&q ...

  2. [置顶] 智能家居开源项目 The open Home Automation Bus (openHAB)

    ================================================================================ 2014-05-19 论文的事情太多, ...

  3. [置顶] Linux 常用命令集锦

    出处:http://www.vaikan.com/what-are-the-most-useful-swiss-army-knife-one-liners-on-unix/ Linux命令行里的&qu ...

  4. 字体样式(.ttf/.woff)文件的配置引入

    在引入前端框架部分功能时,有时需要配置字体样式.可以这样配置:在 .ttf的同级目录下,创建icon.css文件,写入: @font-face {font-family: "element- ...

  5. javaScript(其他引用类型对象)

    javascript其他引用类型对象 Global对象(全局)这个对象不存在,无形的对象,无法new一个 其内部定义了一些方法和属性:如下 encodeURI str = www.baidu.com ...

  6. 每日Linux命令--不完整命令

    配置文件优化,即把默认的空行还有#注释行去掉,优化前先拷贝一份配置文件 egrep -v '^$|#' 拷贝的配置文件 > 原配置文件 mysql如何修改root用户的密码 方法1: 用SET ...

  7. pytorch 4 regression 回归

    import torch import torch.nn.functional as F import matplotlib.pyplot as plt # torch.manual_seed(1) ...

  8. SpringBoot2.0 监听器ApplicationListener的使用-监听ApplicationReadyEvent事件

    参考:http://www.shareniu.com/article/73.htm 一.需求是想将我的写一个方法能在项目启动后就运行,之前使用了redis的消息监听器,感觉可以照着监听器这个思路做,于 ...

  9. 【 henuacm2016级暑期训练-动态规划专题 A 】Cards

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 很显然只要维护B,R,G的数量就好了. 可以很容易想到一个dfs(int a,int b,int c) 然后如果a+b+c==1,那 ...

  10. 集合接口list与集合接口set的区别

    在Java中 除了 Map以外的集合的根接口都是Collection接口,而在Collection接口的子接口中,最重要的莫过于List和Set集合接口. 今天我们就来谈谈List集合接口与Set集合 ...