搜了题解才把题搞明白。明白之后发现其实题意很清晰,解题思路也很清晰,只是题目表述的很不清晰……

大意如下——

给你一个无向图,图中任意两点的距离是两点间所有路径上的某一条边,这条边需要满足两个条件:1. 这条边这两点间某条路径上的最长边;2. 这条边是这两点间所有路径上的最长边中的最短边。

简单来说,假如a到d有两条路径,一条经过b,一条经过d,其中ab = 1, bd = 3, ac = 2, cd = 2,那么abd上的最长边为3,acd上的最长边为2,则ad的距离为2。

如果a, d两点间的距离小于能量L,那么就可以在a, d两点间建立一个传送门。

现在,求在L的能量下最多可以在这个图中建立多少个传送门。

输入:

多组输入数据。

每组输入数据第一行包括三个整数n, m, q。表示节点数,边数,请求数。

接下来m行,每行三个整数u, v, val,表示边的源点,目的点,边权(注意,是无向图,源点和目的点等价)。

接下来q行,每行一个整数L,表示请求所提供的能量。

解题核心:如果集合x与集合y不连通,而此时有一条路L'将x与y连通,且L' <= L,此时将可以建立新传送门num[x]*num[y]个,num[x]表示x集合中的节点数。L1连通后,将集合x与集合y合并,得到新集合x,num[x] += num[y],这就是并查集。

可以使用并查集+kruskal进行求解。即,将所有边从小到大排序,每次按顺序向并查集中增加新边,需要保证添加的新边不会构成环,直到边长>请求所提供的能量L。

新问题出现了,当我们在L1的能量下将路径求出来了,那么如果下一次请求能量为L2,那么我们无法在已有的并查集上继续求解,只能重新建立并查集,这将产生极大的浪费。所以,我们需要将请求L1——Lq全部记录下来,即离线操作,然后按照从小到大的顺序进行求解。最后在将解按照请求顺序排序输出。

上代码——

 #include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
using namespace std; const int M = ; struct Que //保存查询
{
int q, id, ans; //分别是查询值,查询顺序,输出结果
}que[M]; struct Edge //保存边
{
int u, v, val;
}edge[*M]; int fm[M]; //并查集使用
int sum[M]; //记录各区间节点数 int n, m, q; bool cmp(Edge x, Edge y)
{
return x.val <= y.val;
} bool cmp1(Que x, Que y)
{
return x.q <= y.q;
} bool cmp2(Que x, Que y)
{
return x.id < y.id;
} void init()
{
for(int i = ; i <= n; i++)
{
fm[i] = i;
sum[i] = ;
}
for(int i = ; i < m; i++) scanf("%d%d%d", &edge[i].u, &edge[i].v, &edge[i].val);
sort(edge, edge+m, cmp); //按路径长度从小到大排序 for(int i = ; i < q; i++)
{
scanf("%d", &que[i].q);
que[i].id = i;
que[i].ans = ;
}
sort(que, que+q, cmp1); //按请求长度从小到大排序
} int mfind(int x) //查询操作,含路径压缩
{
int fx = x;
while(fx != fm[fx]) fx = fm[fx];
while(x != fm[x])
{
int mid = fm[x];
fm[x] = fx;
x = mid;
}
return fx;
} void work()
{
int cnt = ;
for(int i = ; i < q; i++) //回应请求
{
while(que[i].q >= edge[cnt].val && cnt < m) //kruskal算法
{
int fx = mfind(edge[cnt].u);
int fy = mfind(edge[cnt].v);
if(fx != fy)
{
que[i].ans += sum[fx]*sum[fy]; //新增传送阵
fm[fy] = fx; //集合合并
sum[fx] += sum[fy];
}
cnt++;
}
if(i > ) que[i].ans += que[i-].ans; //包含已有传送阵
}
} void output()
{
sort(que, que+q, cmp2); //按请求顺序排序
for(int i = ; i < q; i++) printf("%d\n", que[i].ans);
} int main()
{
//freopen("test.txt", "r", stdin);
while(~scanf("%d%d%d", &n, &m, &q))
{
init();
work();
output();
}
return ;
}

hdu 3938 Portal(并查集+离线+kruskal)2011 Multi-University Training Contest 10的更多相关文章

  1. HDU 3938 Portal (离线并查集,此题思路很强!!!,得到所谓的距离很巧妙)

    Portal Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Subm ...

  2. ACM: hdu 1811 Rank of Tetris - 拓扑排序-并查集-离线

    hdu 1811 Rank of Tetris Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & % ...

  3. BZOJ5188: [Usaco2018 Jan]MooTube 并查集+离线处理

    BZOJ又不给题面... Luogu的翻译看不下去... 题意简述 有一个$n$个节点的树,边有权值,定义两个节点之间的距离为两点之间的路径上的最小边权 给你$Q$个询问,问你与点$v$的距离超过$k ...

  4. poj 2528 Mayor's posters 线段树 || 并查集 离线处理

    题目链接 题意 用不同颜色的线段覆盖数轴,问最终数轴上有多少种颜色? 注:只有最上面的线段能够被看到:即,如果有一条线段被其他的线段给完全覆盖住,则这个颜色是看不到的. 法一:线段树 按题意按顺序模拟 ...

  5. ACM学习历程—SNNUOJ 1110 传输网络((并查集 && 离线) || (线段树 && 时间戳))(2015陕西省大学生程序设计竞赛D题)

    Description Byteland国家的网络单向传输系统可以被看成是以首都 Bytetown为中心的有向树,一开始只有Bytetown建有基站,所有其他城市的信号都是从Bytetown传输过来的 ...

  6. HDU 3938:Portal(并查集+离线处理)

    http://acm.hdu.edu.cn/showproblem.php?pid=3938 Portal Problem Description   ZLGG found a magic theor ...

  7. zoj3261 并查集离线处理

    Connections in Galaxy War Time Limit:3000MS     Memory Limit:32768KB     64bit IO Format:%lld & ...

  8. HDU 2818 (矢量并查集)

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=2818 题目大意:每次指定一块砖头,移动砖头所在堆到另一堆.查询指定砖头下面有几块砖头. 解题思路: ...

  9. BZOJ-1015 StarWar星球大战 并查集+离线处理

    1015: [JSOI2008]星球大战starwar Time Limit: 3 Sec Memory Limit: 162 MB Submit: 4105 Solved: 1826 [Submit ...

随机推荐

  1. DataRow.RowState 属性

    RowState 的值取决于两个因素:已对该行执行的操作的类型,以及是否已对 DataRow 调用了 AcceptChanges. private void DemonstrateRowState() ...

  2. hdu 1271 整数对

    看了别人的解题报告a了, 大致思路就是 A=a+b*10^k+c*10^(k+1) B=a+c*10^k (在A中取出一位数后) N=A+B=2*a+b*10^k+11*c*10^k 这样就好做了,再 ...

  3. Pycharm中的实用功能(网上看到的,感觉还不错)

    实时比较 PyCharm 对一个文件里你做的改动保持实时的跟踪,通过在编辑器的左侧栏显示一个蓝色的标记.这一点非常方便,我之前一直是在Eclipse里面用命令“Compare against HEAD ...

  4. Oracle 10 - 数据库表

    Oracle数据库表类型 1.堆表 2.索引组织表 3.索引clustered表 4.散列clustered表 5.有序散列clustered表 6.嵌套表 7.临时表 8.对象表 9.外部表 Ora ...

  5. 防止注入网上查了下用SqlParameter可以,那SqlParameter处理单引号时候是自动转义了吗

    String sql = String.Format(); SqlCommand com = new SqlCommand(sql, con); 一般而言,把 一个单引号,替换成 两个单引号就可以了.

  6. Android:实现数组之间的复制

    System提供了一个静态方法arraycopy(),我们可以使用它来实现数组之间的复制 System.arraycopy(src, srcPos, dst, dstPos, length); src ...

  7. PHP之APC缓存详细介绍(转)

    1.APC缓存简介 APC,全称是Alternative PHP Cache,官方翻译叫”可选PHP缓存”.它为我们提供了缓存和优化PHP的中间代码的框架. APC的缓存分两部分:系统缓存和用户数据缓 ...

  8. openfire的配置

    Openfire 采用Java开发,开源的实时协作(RTC)服务器基于XMPP(Jabber)协议.Openfire安装和使用都非常简单,并利用Web进行管理.单台服务器可支持上万并发用户.所以常常被 ...

  9. C#获取网页中某个元素的位置,并模拟点击

    我们在开发中,往往要得到网页中某个元素的位置,并且点击它.要模拟一次鼠标点击并不难,只要调用一个API就行了,关键就是怎么样得到这个元素的位置,还有判断是否要滚动滚动条,要滚动多少行能让元素显示出来. ...

  10. java.lang.NoClassDefFoundError: javax/wsdl/OperationType

    You should find the javax.wsdl package inside wsdl4j.jar Check for the line starting with 'Found IBM ...