Description

N-1座桥连接着N个岛屿,每座桥都连接着某两个不同的岛屿,从任意一个岛屿都可以到达所有的其他岛屿,过桥需要缴纳人民币1元的过桥费。

由于某些不可透露的原因,Jason和他的2个小伙伴可以在任意一个岛屿集合,但是希望总过桥费最少。

现在,由你来确定集合的岛屿编号,使得总过桥费最少。

Input Format

第一行有两个整数N和M,表示岛屿的个数和询问次数,岛屿编号从1到N。

接下来N-1行,每行有两个正整数X、Y,表示编号为X的岛屿和编号为Y的岛屿之间有一座桥。

最后还有M行,每行有三个正整数A、B、C,表示Jason和他的两个小伙伴所在岛屿的编号。

Output Format

一共有M行,每行两个整数P、Q,用一个空格隔开。其中第i行表示第i次询问集合的地点在编号为P的岛屿,需要花费的最少过桥费为Q。

Sample Output

5 5

1 2

1 3

2 4

2 5

1 1 1

1 1 2

1 2 3

4 5 3

5 2 3

Sample Output

1 0

1 1

1 2

2 4

2 3

Hint

30%的数据中,N≤200,M≤200;

60%的数据中,N≤2,000,M≤2,000;

100%的数据中,N≤200,000,M≤200,000;

Solution

显然这是一个树形结构,可以发现,3个地点A,B,C,一定存在2对的LCA相同,例如LCA(A,B)==LCA(A,C),那么集合点就在LCA(B,C),画一画图会更好理解

还有几种特殊的情况经检验也符合上述情况

Code

#include <cstdio>
#include <algorithm>
#include <cmath>
#define N 200010
#define M 400010 struct info {int to, nex;} e[M];
int n, m, tot, head[M], fa[N][22], dep[N], _log;
bool vis[N]; void dfs(int u) {
vis[u] = 1;
for (int i = 1; dep[u] >= (1 << i); ++i)
fa[u][i] = fa[fa[u][i - 1]][i - 1]; for (int i = head[u]; i; i = e[i].nex) {
int v = e[i].to;
if (vis[v]) continue;
dep[v] = dep[u] + 1;
fa[v][0] = u;
dfs(v);
}
} inline int read() {
int x = 0, f = 1; char ch = getchar();
while (ch < '0' || ch > '9') {if (ch == '-')f = -1; ch = getchar();}
while (ch >= '0' && ch <= '9') {x = x * 10 + ch - '0'; ch = getchar();}
return x * f;
} inline void add_edge(int u, int v) {
e[++tot].to = v;
e[tot].nex = head[u];
head[u] = tot;
} int LCA(int u, int v) {
if (dep[u] > dep[v])
std::swap(u, v); int d = dep[v] - dep[u];
for (int i = 0; i <= _log; ++i)
if (d & (1 << i))
v = fa[v][i]; if (u == v) return u; for (int i = _log; i >= 0; i--)
if (fa[u][i] != fa[v][i]) {
u = fa[u][i];
v = fa[v][i];
} return fa[u][0];
} inline void work(int a, int b, int c) {
int ab = LCA(a, b);
int ans = dep[a] + dep[b] - 2 * dep[ab] + dep[ab] + dep[c] - 2 * dep[LCA(ab, c)];
printf("%d %d\n", ab, ans);
} int main() {
n = read(), m = read();
_log = log(n) / log(2);
for (int i = 1; i < n; ++i) {
int u = read(), v = read();
add_edge(u, v);
add_edge(v, u);
}
dfs(1); while (m--) {
int a = read(), b = read(), c = read();
int ab = LCA(a, b), ac = LCA(a, c), bc = LCA(b, c);
if (ac == bc) work(a, b, c);
else if (ab == bc) work(a, c, b);
else if (ab == ac) work(b, c, a);
}
return 0;
}

旅行(LCA)的更多相关文章

  1. CodeVs.1036 商务旅行 ( LCA 最近公共祖先 )

    CodeVs.1036 商务旅行 ( LCA 最近公共祖先 ) 题意分析 某首都城市的商人要经常到各城镇去做生意,他们按自己的路线去做,目的是为了更好的节约时间. 假设有N个城镇,首都编号为1,商人从 ...

  2. 【codevs1036】商务旅行 LCA 倍增

    1036 商务旅行  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond 题目描述 Description 某首都城市的商人要经常到各城镇去做生意,他们按自己的 ...

  3. [codevs1036]商务旅行<LCA:tarjan&倍增>

    题目链接:http://codevs.cn/problem/1036/ 今天翻箱倒柜的把这题翻出来做了,以前做的时候没怎么理解,所以今天来重做一下 这题是一个LCA裸题,基本上就把另一道裸题小机房的树 ...

  4. 倍增法-lca codevs 1036 商务旅行

    codevs 1036 商务旅行  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond 题目描述 Description 某首都城市的商人要经常到各城镇去做生意 ...

  5. codevs 1036 商务旅行 (倍增LCA)

    /* 在我还不知道LCA之前 暴力跑的SPFA 70分 三个点TLE */ #include<iostream> #include<cstdio> #include<cs ...

  6. BZOJ2746: [HEOI2012]旅行问题(AC自动机 LCA)

    Time Limit: 30 Sec  Memory Limit: 256 MBSubmit: 1188  Solved: 383[Submit][Status][Discuss] Descripti ...

  7. codevs1036商务旅行(LCA)

    1036 商务旅行  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond 题解       题目描述 Description 某首都城市的商人要经常到各城镇去做 ...

  8. cogs 186. [USACO Oct08] 牧场旅行 树链剖分 LCA

    186. [USACO Oct08] 牧场旅行 ★★☆   输入文件:pwalk.in   输出文件:pwalk.out   逐字节对比时间限制:1 s   内存限制:128 MB n个被自然地编号为 ...

  9. codevs 1036 商务旅行(Targin求LCA)

    传送门 Description 某首都城市的商人要经常到各城镇去做生意,他们按自己的路线去做,目的是为了更好的节约时间. 假设有N个城镇,首都编号为1,商人从首都出发,其他各城镇之间都有道路连接,任意 ...

随机推荐

  1. 201521123115《Java程序设计》第2周学习总结

    1. 本章学习总结 Arrays和String的用法及其函数的一些运用,例如sort函数,输入多个数字,用Arrays.sort(数组名),对数组中的元素排序 2. 书面作业 **Q1.使用Eclip ...

  2. Java 第十四周总结

    1. 本周学习总结 2. 书面作业 1. MySQL数据库基本操作 2. 使用JDBC连接数据库与Statement 2.1 使用Statement操作数据库.(粘贴一段你认为比较有价值的代码,出现学 ...

  3. eclipse: eclipse创建java web项目

    Eclipse创建java web工程 eclipse版本:eclipse-jee-4.5-win32-x64 tomcat版本:apache-tomcat-7.0.63-windows-x64 jd ...

  4. Thinkphp3.2结合phpqrcode生成二维码(含Logo的二维码),附案例

    首先,下载phpqrcode,将其解压到项目ThinkPHP\Library\Vendor目录下.Index_index.html(模板可自行配置) <form action="{:U ...

  5. mysql死锁+解决

    自己作死,navicat不恰当的操作导致了表死锁,操作如下: 给表新加字段:name 没有选择允许为空,但是有没有设置初始值,所以运行的结果就是数据库表里有了name不允许为空但是确实为空的记录: 然 ...

  6. arm-linux-gcc 4.3.2编译uboot 1.1.6

    在第三期项目的视频中,官方提供了一整套新的工具链,bootloader, 内核和文件系统(arm-linux-gcc_4.3.2, uboot-2012.04.01, linux-3.4.2)其中ub ...

  7. Jmeter运行结果分析

    1.聚合报告 以上就是我们所关心的结果 Label:页面或请求名称 Samples:运行的线程数(也可理解为请求数) Average:平均响应时间 Median:响应时间的中值 90% Line:90 ...

  8. 西邮linux兴趣小组2014纳新免试题(二)

    [第二关] 题目 http://round2.sinaapp.com/ 分析 打开后,戳进去发现一句名言,然后下一戳的url提示. 在网页源码中得到Page1024提示,于是写一个脚本 #!/bin/ ...

  9. MySQL高级查询(一)

    修改表 修改表名 语法: ALTER  TABLE<旧表名> RENAME  [TO] <新表名>; 添加字段 语法: ALTER  TABLE 表名 ADD 字段名 数据类型 ...

  10. yum软件管理器,及yum源配置

    说到yum源就必须说到linux系统中特有的依赖关系问题,yum就是为了解决依赖关系而存在的.yum源就相当是一个目录项,当我们使用yum机制安装软件时,若需要安装依赖软件,则yum机制就会根据在yu ...