[SPOJ913]QTREE2 - Query on a tree II【倍增LCA】
题目描述
题目大意
给一棵树,有两种操作:
- 求(u,v)路径的距离。
- 求以u为起点,v为终点的第k的节点.
分析
比较简单的倍增LCA模板题。
首先对于第一问,我们只需要预处理出根节点到各个节点之间的距离,然后倍增LCA求解就可以了。
那么第二问我WA了6发,原来是眼瞎和手残打错了两个字符错掉了。
我们将问题分成3个部分:
- LCA是第k个
- 第k个在u到LCA的路径上
- 第k个在LCA到v的路径上。
首先如果LCA是第k个,那么直接输出。
如果是第二种情况,那么从u开始做倍增,每一次k-(1<<i)就可以了。
小细节:只能将k变成1,模拟可证明。
第三种情况,那么我们的答案就是从v的第(dep[u]+dep[v]-2*dep[lca]+2),模拟可得该式。
ac代码
#include <bits/stdc++.h>
#define ll long long
#define ms(a, b) memset(a, b, sizeof(a))
#define inf 0x3f3f3f3f
#define ll long long
using namespace std;
template <typename T>
inline void read(T &x) {
x = 0; T fl = 1;
char ch = 0;
while (ch < '0' || ch > '9') {
if (ch == '-') fl = -1;
ch = getchar();
}
while (ch >= '0' && ch <= '9') {
x = (x << 1) + (x << 3) + (ch ^ 48);
ch = getchar();
}
x *= fl;
}
#define N 20005
struct edge {
int to, nt;
ll w;
}E[N << 1];
ll dis[N];
int f[N][31], H[N], dep[N];
int cnt, n;
void add_edge(int u, int v, ll w) {
E[++ cnt] = (edge){v, H[u], w};
H[u] = cnt;
}
void dfs(int u, int fa, int dist) {
f[u][0] = fa;
dis[u] = dist;
for (int i = 1; i <= 30; i ++)
f[u][i] = f[f[u][i - 1]][i - 1];
for (int e = H[u]; e; e = E[e].nt) {
int v = E[e].to;
if (v == fa) continue;
dep[v] = dep[u] + 1;
dfs(v, u, dist + E[e].w);
}
}
int lca(int u, int v) {
if (dep[u] < dep[v]) swap(u, v);
for (int i = 30; i >= 0; i --)
if (dep[v] <= dep[f[u][i]]) u = f[u][i];
if (u == v) return u;
for (int i = 30; i >= 0; i --)
if (f[u][i] != f[v][i]) {
u = f[u][i];
v = f[v][i];
}
return f[u][0];
}
int solve(int u, int v, int k) {
int Lca = lca(u, v);
if (dep[u] - dep[Lca] + 1 == k) return Lca;
else {
if (dep[u] - dep[Lca] + 1 > k) {
for (int i = 30; i >= 0; i --)
if (k - 1 >= (1 << i)) k -= (1 << i), u = f[u][i];
return u;
}
else {
k = dep[u] + dep[v]- dep[Lca] * 2 - k + 2;
for (int i = 30; i >= 0; i --)
if (k - 1 >= (1 << i)) k -= (1 << i), v = f[v][i];
return v;
}
}
}
int main() {
int cas;
read(cas);
char opt[10];
while (cas --) {
cnt = 0;
ms(H, 0);
ms(dis, 0);
ms(dep, 0);
ms(f, 0);
read(n);
for (int i = 1; i < n; i ++) {
int u, v; ll w;
read(u); read(v); read(w);
add_edge(u, v, w);
add_edge(v, u, w);
}
dep[1] = 1;
dfs(1, 0, 0);
while (1) {
scanf("%s", opt);
if (opt[1] == 'O') break;
if (opt[0] == 'D') {
int u, v;
read(u); read(v);
int Lca = lca(u, v), res;
printf("%lld\n", dis[u] + dis[v] - dis[Lca] * 2);
}
else {
int u, v, k;
read(u); read(v); read(k);
printf("%d\n", solve(u, v, k));
}
}
}
return 0;
}
[SPOJ913]QTREE2 - Query on a tree II【倍增LCA】的更多相关文章
- spoj 913 Query on a tree II (倍增lca)
Query on a tree II You are given a tree (an undirected acyclic connected graph) with N nodes, and ed ...
- 【SPOJ QTREE2】QTREE2 - Query on a tree II(LCA)
You are given a tree (an undirected acyclic connected graph) with N nodes, and edges numbered 1, 2, ...
- Query on a tree II 倍增LCA
You are given a tree (an undirected acyclic connected graph) with N nodes, and edges numbered 1, 2, ...
- LCA SP913 QTREE2 - Query on a tree II
SP913 QTREE2 - Query on a tree II 给定一棵n个点的树,边具有边权.要求作以下操作: DIST a b 询问点a至点b路径上的边权之和 KTH a b k 询问点a至点 ...
- SP913 QTREE2 - Query on a tree II
思路 第一个可以倍增,第二个讨论在a到lca的路径上还是lca到b的路径上, 倍增即可 代码 #include <cstdio> #include <algorithm> #i ...
- SPOJ QTREE2 Query on a tree II
传送门 倍增水题…… 本来还想用LCT做的……然后发现根本不需要 //minamoto #include<bits/stdc++.h> using namespace std; #defi ...
- SPOJ COT2 - Count on a tree II(LCA+离散化+树上莫队)
COT2 - Count on a tree II #tree You are given a tree with N nodes. The tree nodes are numbered from ...
- QTREE2 spoj 913. Query on a tree II 经典的倍增思想
QTREE2 经典的倍增思想 题目: 给出一棵树,求: 1.两点之间距离. 2.从节点x到节点y最短路径上第k个节点的编号. 分析: 第一问的话,随便以一个节点为根,求得其他节点到根的距离,然后对于每 ...
- SPOJ913 Query on a tree II
Time Limit: 433MS Memory Limit: 1572864KB 64bit IO Format: %lld & %llu Description You are g ...
随机推荐
- Python_数据类型的补充、集合set、深浅copy
1.数据类型的补充 1.1 元组 当元组里面只有一个元素且没有逗号时,则该数据的数据类型与括号里面的元素相同. tu1 = ('laonanhai') tu2 = ('laonanhai') prin ...
- JEECG & JEESite Tomcat集群 Session共享
多台tomcat服务的session共享 memcached与redis - JEECG开源社区 - CSDN博客https://blog.csdn.net/zhangdaiscott/article ...
- Nginx三部曲(1)基础
我们会告诉你 Nginx 是如何工作的,其背后的概念有哪些,以及如何优化它以提升应用程序的性能.还会告诉你如何安装,如何启动.运行. 这个教程包括三节: 基础概念——你可以了解命令(directive ...
- js-其他跨域技术(JSONP`Comet)
###1. JSONP JSONP由两部分组成:回调函数和数据 JSONP是通过动态<script>元素来使用的,使用时可以为src属性指定一个跨域URL eg: function ha ...
- JavaScript生成二维码图片
1.引入一个二维码工具的js文件,同时需要引入jquery文件 下面是jquery.qrcode.min.js文件内容: (function(r){r.fn.qrcode=function(h){va ...
- package-lock.json和package.json的作用
转自:https://www.cnblogs.com/cangqinglang/p/8336754.html package-lock.json的作用就是锁定安装依赖时包的版本,并且需要上传到git, ...
- 理解git工作区和暂存区
版本库 在工作区目录中有一个.git文件,这个其实不是工作区而是Git的版本库 版本库中包含两个部分,一个是暂存区index/stage,另一个是git自动为我们创建的第一个分支master,以及一个 ...
- MSDN学习: 加密解密Config文件中的Sections( Encrypting and Decrypting Configuration Sections)
https://msdn.microsoft.com/en-us/library/wfc2t3az(v=vs.100).aspx https://msdn.microsoft.com/en-us/li ...
- qtp 自动化测试--点滴 菜单没有了,有些控件运行时找不到
test项目页签下-没有了 菜单栏:file edit view insert 看不到了 1 解决:在startpage标签下-tool-option-点击 restore layout-确定 2 菜 ...
- vue之综合Demo:打沙袋
demo7.html <!DOCTYPE html> <html lang="en" xmlns:v-bind="http://www.w3.org/1 ...