题目链接 \(Click\) \(Here\)

\(Kruskal\)重构树的好题。想到的话就很好写,想不到乱搞的难度反而相当高。

按照点的水位,建出来满足小根队性质的\(Kruskal\)重构树,这样一个点的子树里的点就是所有可以开车到达的点。做一遍最短路预处理,然后树上求一个子树\(min\),就可以得到子树里面的点到点\(1\)的最短距离。注意需要初始化。

#include <bits/stdc++.h>
using namespace std; const int N = 800010;
const int INF = 0x7fffffff; struct _edge {int u, v, l, a;}_e[N]; bool cmp (_edge lhs, _edge rhs) {
return lhs.a > rhs.a;
} struct Graph {
int cnt, head[N]; struct edge {
int nxt, to, w;
}e[N << 1]; void Init () {
cnt = 0;
memset (head, 0, sizeof (head));
} void add_edge (int u, int v, int w) {
e[++cnt] = (edge) {head[u], v, w}; head[u] = cnt;
}
}G, krus; int read () {
int s = 0, w = 1, ch = getchar ();
while ('9' < ch || ch < '0') {
if (ch == '-') w = -1;
ch = getchar ();
}
while ('0' <= ch && ch <= '9') {
s = s * 10 + ch - '0';
ch = getchar ();
}
return s * w;
} int T, n, m, Q, K, S, tot, fa[N], _high[N]; int find (int x) {
return fa[x] == x ? x : fa[x] = find (fa[x]);
} int deep[N], fafa[N][20]; int dis[N], mindis[N]; void dfs (int u, int fa) {
fafa[u][0] = fa;
mindis[u] = krus.head[u] == 0 ? dis[u] : INF;
deep[u] = deep[fa] + 1;
for (int i = 1; (1 << i) <= deep[u]; ++i) {
fafa[u][i] = fafa[fafa[u][i - 1]][i - 1];
}
for (int i = krus.head[u]; i; i = krus.e[i].nxt) {
int v = krus.e[i].to;
dfs (v, u);
mindis[u] = min (mindis[u], mindis[v]);
}
} void kruskal () {
sort (_e + 1, _e + 1 + m, cmp);
tot = n;
for (int i = 1; i <= n; ++i) fa[i] = i;
for (int i = 1; i <= m; ++i) {
int u = find (_e[i].u);
int v = find (_e[i].v);
if (u != v) {
int T = ++tot;
_high[T] = _e[i].a;
krus.add_edge (T, u, 0);
krus.add_edge (T, v, 0);
fa[T] = fa[u] = fa[v] = T;
}
}
dfs (tot, 0);
} struct Node {
int pos, dis; bool operator < (Node rhs) const {
return dis > rhs.dis;
}
}; priority_queue <Node> q; void dijkstra () {
for (int i = 1; i <= n; ++i) dis[i] = i == 1 ? 0 : INF;
q.push ((Node) {1, 0});
while (!q.empty ()) {
Node u = q.top (); q.pop ();
if (dis[u.pos] < u.dis) continue;
for (int i = G.head[u.pos]; i; i = G.e[i].nxt) {
int v = G.e[i].to;
if (dis[v] > dis[u.pos] + G.e[i].w) {
dis[v] = dis[u.pos] + G.e[i].w;
q.push ((Node) {v, dis[v]});
}
}
}
} int query (int u, int p) {
//u 出发节点 p 水位线
for (int i = 19; i >= 0; --i) {
if (_high[fafa[u][i]] > p) {
u = fafa[u][i];
}
}
// printf ("u = %d\n", u);
return mindis[u];
} void Init () {
G.Init ();
krus.Init ();
memset (fafa, 0, sizeof (fafa));
memset (_high, 0, sizeof (_high));
memset (mindis, 0, sizeof (mindis));
} int main () {
//freopen ("data.in", "r", stdin);
T = read ();
while (T--) {
Init ();
printf ("T = %d\n", T);
n = read (), m = read ();
for (int i = 1; i <= m; ++i) {
_e[i].u = read ();
_e[i].v = read ();
_e[i].l = read ();
_e[i].a = read ();
G.add_edge (_e[i].u, _e[i].v, _e[i].l);
G.add_edge (_e[i].v, _e[i].u, _e[i].l); //建双向边
}
int lastans = 0;
dijkstra ();
kruskal ();
Q = read (), K = read (), S = read ();
for (int i = 1; i <= Q; ++i) {
static int v, p, v0, p0;
v0 = read (), p0 = read ();
v = (v0 + K * lastans - 1) % n + 1;
p = (p0 + K * lastans) % (S + 1);
printf ("%d\n", lastans = query (v, p));
}
}
}

Luogu P4768 [NOI2018]归程的更多相关文章

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

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

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

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

  3. [洛谷P4768] [NOI2018]归程 (kruskal重构树模板讲解)

    洛谷题目链接:[NOI2018]归程 因为题面复制过来有点炸格式,所以要看题目就点一下链接吧\(qwq\) 题意: 在一张无向图上,每一条边都有一个长度和海拔高度,小\(Y\)的家在\(1\)节点,并 ...

  4. 洛谷P4768 [NOI2018]归程 [可持久化并查集,Dijkstra]

    题目传送门 归程 格式难调,题面就不放了. 分析: 之前同步赛的时候反正就一脸懵逼,然后场场暴力大战,现在呢,还是不会$Kruskal$重构树,于是就拿可持久化并查集做. 但是之前做可持久化并查集的时 ...

  5. P4768 [NOI2018]归程

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

  6. 洛谷P4768 [NOI2018]归程(Kruskal重构树)

    题意 直接看题目吧,不好描述 Sol 考虑暴力做法 首先预处理出从$1$到每个节点的最短路, 对于每次询问,暴力的从这个点BFS,从能走到的点里面取$min$ 考虑如何优化,这里要用到Kruskal重 ...

  7. 洛谷P4768 [NOI2018]归程(可持久化并查集,最短路)

    闲话 一个蒟蒻,在网络同步赛上进行了这样的表演-- T2组合计数不会,T3字符串数据结构不会,于是爆肝T1 一开始以为整个地图都有车,然后写了2h+的树套树,终于发现样例过不去 然后写可持久化并查集D ...

  8. 洛谷 P4768 [NOI2018]归程

    洛谷 361行代码的由来 数据分治大发好啊- NOI的签到题,可怜我在家打了一下午才搞了80分. 正解应该是kruskal重构树或排序+可持久化并查集. 我就分点来讲暴力80分做法吧(毕竟正解我也没太 ...

  9. Luogu 4768 [NOI2018]归程

    并不会写Kruskal重构树,两个$log$跑得比较卡. 首先考虑一下没有强制在线的要求怎么办,有一个比较容易想到的做法就是先跑一遍最短路,然后把所有边按照海拔从大到小排序,把所有询问的海拔也从大到小 ...

随机推荐

  1. 转 在PowerDesigner的PDM图形窗口中显示数据列的中文注释

    Name是名称(字段描述),Code是字段名称,Comment是注释名称,ER图中显示的是Name.一般设计时,Name跟comment都设计成描述, 而设计时候常把comment写成中文,name保 ...

  2. 为WebRTC 应用部署Turn Server

    部署WebRTC 或 SIP p2p 方案时经常会遇到p2p 无法穿透的环境, 这时就是TunServer 的用武之地了. 这里我们使用turnserver-0.7.3 下载confuse依赖库 wg ...

  3. josn的格式化

    public String formatJson(Object obj) { com.alibaba.fastjson.JSONObject json=(com.alibaba.fastjson.JS ...

  4. Web API 2 自定义默认Identity Table Name

    One of the first issues you will likely encounter when getting started with ASP.NET Identity centers ...

  5. Redis之父表示ARM服务器没戏!

    ARM表示Neoverse N1平台和E1 CPU即将发布,Neoverse N1和E1采用7nm制程,并且为服务器和通信设备增加重要提升,拥有高可扩展性.高处理量以及高性能,将分别在2020年和20 ...

  6. 人工智能将继续壮大,两会委员建议增加“AI+教育”支持板块

    导读 今年上海两会期间,上海市政协委员.上海交通大学机械与动力工程学院教授范秀敏提交提案,建议政府在推进上海人工智能专项建设中,增加“AI+教育”专项支持板块,并鼓励集聚发展AI产业的各个区,在人工智 ...

  7. Java启动命令与Maven打包设置

    一.Java启动命令 java程序的启动方式有三种: 1.java -jar 生成的jar包中,manifest文件定义了Main Class,可使用该命令 java -jar test.jar 2. ...

  8. HTML5-Input

    HTML5拥有多个新的表单输入类型,这些新特性提供了更好的输入控制和验证(有的浏览器不支持) color.date.datetime.datetime-local.email.month.number ...

  9. 「Splay」区间翻转

    传送门:>Here< 解法分析 用splay来维护这个序列. 一直没有搞明白的是,这里的splay的节点究竟维护的是什么?是权值吗?肯定不是,因为区间是会翻转的,如果维护权值的话很快平衡树 ...

  10. Bomb Game HDU - 3622(二分最小值最大化)

    题意: 就是给出n对坐标,每对只能选一个,以选出来的点为圆心,半径自定义,画圆,而这些圆不能覆盖,求半径最小的圆的最大值 解析: 看到最x值最x化,那二分变为判定性问题,然后...然后我就没想到... ...