题解

Kruskal重构树:每次一条边连接两个集合,建一个新点,点权为该边边权;把这两个集合的根连向新点。

性质:(如果求的是最大生成树)叶子结点是图中实际结点;叶子到根路径上点权递减;两点间lca的权值就是这两点走最大生成树经过的最小边

然后对于这题我们建重构树然后每次倍增找到一个深度极小的祖先u,返回u子树内实结点dis的最小值即可

#include <algorithm>
#include <cstdio>
#include <queue>
using namespace std;
typedef long long ll;
const int N = 4e5 + 10;
const ll INF = 1e15;
int n, m, q, k, s;
struct Edge {
int v, w, nxt;
} e[N * 2];
int hd[N], c;
ll d[N];
void add(int u, int v, int w) {
e[c] = (Edge) {v, w, hd[u]}; hd[u] = c ++;
}
struct Node {
int u; ll d;
bool operator < (const Node &b) const { return d > b.d; }
};
void dijkstra() {
fill(d + 1, d + n + 1, INF); d[1] = 0;
priority_queue<Node> pq; pq.push((Node) {1, d[1]});
while(pq.size()) {
Node k = pq.top(); pq.pop();
if(d[k.u] < k.d) continue ;
for(int i = hd[k.u]; ~ i; i = e[i].nxt) {
if(d[e[i].v] > d[k.u] + e[i].w) {
pq.push((Node) {e[i].v, d[e[i].v] = d[k.u] + e[i].w});
}
}
}
}
struct Edge2 {
int u, v, w;
bool operator < (const Edge2 &e) const { return w > e.w; }
} e2[N];
int f[N], T[N], p[N][22], w[N], nn;
ll dt[N];
int find(int u) { return u == f[u] ? u : f[u] = find(f[u]); }
void kruskal() {
sort(e2 + 1, e2 + m + 1);
for(int i = 1; i <= n; i ++) f[i] = T[i] = i, dt[i] = d[i];
int cnt = 0; nn = n;
for(int i = 1; i <= m; i ++) {
int u = find(e2[i].u), v = find(e2[i].v);
if(u != v) {
p[T[u]][0] = p[T[v]][0] = ++ nn;
w[nn] = e2[i].w; p[nn][0] = 0;
dt[nn] = min(dt[T[u]], dt[T[v]]);
f[u] = v; T[v] = nn;
if(++ cnt == n - 1) break ;
}
}
for(int j = 1; j <= 20; j ++)
for(int i = 1; i <= nn; i ++)
p[i][j] = p[p[i][j - 1]][j - 1];
} ll qry(int u, int low) {
int v = u;
for(int i = 20; i >= 0; i --) {
if(p[v][i] && w[p[v][i]] > low) {
v = p[v][i];
}
}
return dt[v];
} void solve() {
scanf("%d%d", &n, &m);
fill(hd + 1, hd + n + 1, -1); c = 0;
for(int i = 1; i <= m; i ++) {
int u, v, l, h;
scanf("%d%d%d%d", &u, &v, &l, &h);
add(u, v, l); add(v, u, l);
e2[i] = (Edge2) {u, v, h};
}
dijkstra();
kruskal();
scanf("%d%d%d", &q, &k, &s);
ll lans = 0;
for(int i = 1; i <= q; i ++) {
int v, w; scanf("%d%d", &v, &w);
v = (v + k * lans - 1) % n + 1;
w = (w + k * lans) % (s + 1);
printf("%lld\n", lans = qry(v, w));
}
} int main() {
freopen("return.in", "r", stdin);
freopen("return.out", "w", stdout);
int T; scanf("%d", &T);
while(T --) solve();
fclose(stdin); fclose(stdout);
return 0;
}

「NOI 2018」归程「Kruskal 重构树」的更多相关文章

  1. 【NOI 2018】归程(Kruskal重构树)

    题面在这里就不放了. 同步赛在做这个题的时候,心里有点纠结,很容易想到离线的做法,将边和询问一起按水位线排序,模拟水位下降,维护当前的各个联通块中距离$1$最近的距离,每次遇到询问时输出所在联通块的信 ...

  2. [luogu4768] [NOI2018] 归程 (Dijkstra+Kruskal重构树)

    [luogu4768] [NOI2018] 归程 (Dijkstra+Kruskal重构树) 题面 题面较长,这里就不贴了 分析 看到不能经过有积水的边,即不能经过边权小于一定值的边,我们想到了kru ...

  3. P4768 [NOI2018]归程(kruskal 重构树)

    洛谷P4768 [NOI2018]归程 LOJ#2718.「NOI2018」归程 用到 kruskal 重构树,所以先说这是个啥 显然,这和 kruskal 算法有关系 (废话 这个重构树是一个有点权 ...

  4. NOI2018归程(Kruskal重构树)

    题目描述 本题的故事发生在魔力之都,在这里我们将为你介绍一些必要的设定. 魔力之都可以抽象成一个 n 个节点.m 条边的无向连通图(节点的编号从 1 至 n). 我们依次用 l,a 描述一条边的长度. ...

  5. Luogu P4768 [NOI2018]归程(Dijkstra+Kruskal重构树)

    P4768 [NOI2018]归程 题面 题目描述 本题的故事发生在魔力之都,在这里我们将为你介绍一些必要的设定. 魔力之都可以抽象成一个 \(n\) 个节点. \(m\) 条边的无向连通图(节点的编 ...

  6. 【BZOJ5415&UOJ393】归程(Kruskal重构树,最短路)

    题意:From https://www.cnblogs.com/Memory-of-winter/p/11628351.html 思路:先从1开始跑一遍dijkstra,建出kruskal重构树之后每 ...

  7. NOI2018 D1T1 洛谷P4768 归程 (Kruskal重构树)

    实际上是一个最短路问题,但加上了海拔这个条件限制,要在海拔<水位线p中找最短路. 这里使用Kruskal重构树,将其按海拔建成小根堆,我们就可以在树中用倍增找出他不得不下车的点:树中节点有两个权 ...

  8. NOI2018d1t1 归程 (dijkstra+kruskal重构树)

    题意:给一张无向联通图,每条边有长度和高度,每次询问在高度大于p的边,从v点能到达的所有点到1号点的最短距离(强制在线) 首先dijkstra求出每个点到1号点的距离 易知:如果我按高度从高到低给边排 ...

  9. LOJ #2718. 「NOI2018」归程(Dijkstra + Kruskal重构树 + 倍增)

    题意 给你一个无向图,其中每条边有两个值 \(l, a\) 代表一条边的长度和海拔. 其中有 \(q\) 次询问(强制在线),每次询问给你两个参数 \(v, p\) ,表示在 \(v\) 出发,能开车 ...

随机推荐

  1. 剑指Offer(4)——替换空格

    题目: 请实现一个函数,把字符串中的每个空格替换成"%20".例如输入“We are happy.”,则输出“We%20are%20happy.”. 思路: 如果按照顺序从前往后依 ...

  2. IdentityServer3 使用记录

    官方教程:https://identityserver.github.io/Documentation/docsv2/overview/mvcGettingStarted.html 1.是否启用 SS ...

  3. 获取类的描述信息 DescriptionAttribute

    static void Main(string[] args) { var attrs = typeof(TestClass).GetCustomAttributes(typeof(System.Co ...

  4. 树节点递归删除--service层

    @Service public class ContentCategoryServiceImpl extends BaseServiceImpl<ContentCategory> impl ...

  5. 基于【 centos7】一 || 安装ELK

    一.安装jdk 上传安装包并解压:tar -zxvf ... 配置环境变量: 在配置文件中添加如下配置信息:vi /etc/profile export JAVA_HOME=/usr/local/jd ...

  6. Javascript--HTML DOM基础知识

    1.HTML DOM是什么,以及它的作用: w3c对DOM有一系列的解释和定义,用自己理解的话来说就是:HTML DOM是html的标准对象模型,可以使JavaScript去操作(获取,修改,删除,添 ...

  7. 美团Java工程师面试题(2018秋招)

    第一次面试 1.小数是怎么存的 2.算法题:N二进制有多少个1 3.Linux命令(不熟悉 4.JVM垃圾回收算法 5.C或者伪代码实现复制算法 6.volatile 7.树的先序中序后序以及应用场景 ...

  8. 怎么读取properties文件和ini文件?

    一.读取properties文件: properties中的内容: server.ip = 127.0.0.1 server.port = 22 //原生java即可读取public static v ...

  9. springboot系列(三) 启动类中关键注解作用解析

    一.Springboot:请求入口 @SpringBootApplication @EnableAspectJAutoProxy @EnableScheduling @EnableTransactio ...

  10. SRX550路由器缓存满了无法在web页面操作解决方法

    SRX550路由器缓存满了无法在web页面操作解决方法   首页出现下图为满的标志,我这个文档就是解决这中情况,让web页面可以操作的 1.  打开命令行,输入用户密码,进入路由器 注意:这里使用te ...