Problem
在业余时间,Farmer John创建了一个新的视频共享服务,他将其命名为MooTube。在MooTube上,Farmer John的奶牛可以录制,分享和发现许多有趣的视频。他的奶牛已经发布了 NN 个视频 ( 1 \leq N \leq 100,0001≤N≤100,000 ),为了方便将其编号为 1 \ldots N1…N 。然而,FJ无法弄清楚如何帮助他的奶牛找到他们可能喜欢的新视频。
FJ希望为每个MooTube视频创建一个“推荐视频”列表。这样,奶牛将被推荐与他们已经观看过的视频最相关的视频。
FJ设计了一个“相关性”度量标准,顾名思义,它确定了两个视频相互之间的相关性。他选择 N-1N−1 对视频并手动计算其之间的相关性。然后,FJ将他的视频建成一棵树,其中每个视频是节点,并且他手动将 N-1N−1 对视频连接。为了方便,FJ选择了 N-1N−1 对,这样任意视频都可以通过一条连通路径到达任意其他视频。 FJ决定将任意一对视频的相关性定义为沿此路径的任何连接的最小相关性。
Farmer John想要选择一个 KK 值,以便在任何给定的MooTube视频旁边,推荐所有其他与该视频至少有 KK 相关的视频。然而,FJ担心会向他的奶牛推荐太多的视频,这可能会分散他们对产奶的注意力!因此,他想设定适当的 KK 值。 Farmer John希望得到您的帮助,回答有关 KK 值的推荐视频的一些问题。
Solution
这道题最简单的方法就是预处理所有的两点间的最小值,但是时间复杂度 O ( N 2 ) O(N^2) O(N2) 显然不可取。
我们可以采用离线的方式来解决这一个问题。
对于每一个询问的k,我们知道如果要满足这个题目中的条件,经过的任何一条路径都要大于或等于k;此时我们可以用并查集来维护这一个关系,把左右边权大于或等于k的所有点都合并为一个集合;这样就表示:这个集合内的每一个点都可以通过大于或等于k的路径之间或间接到达。
对于询问每一个点,答案就是这个点所处的并查集大小-1.
由于询问的k要把大于或等于k的边合并,而k更小的时候,合并的边一定包括更大的k时合并的边;因此我们可以对于每一个循环的k从大到小排序。
好玄学的方法。。。
#include<bits/stdc++.h>
using namespace std;
int N,Q,tot=0,cnt=0;
struct edge
{
int x,y,v;
}a[200000];
struct ques
{
int k,v,num