题目描述 Description

小机房有棵焕狗种的树,树上有N个节点,节点标号为0到N-1,有两只虫子名叫飘狗和大吉狗,分居在两个不同的节点上。有一天,他们想爬到一个节点上去搞基,但是作为两只虫子,他们不想花费太多精力。已知从某个节点爬到其父亲节点要花费 c 的能量(从父亲节点爬到此节点也相同),他们想找出一条花费精力最短的路,以使得搞基的时候精力旺盛,他们找到你要你设计一个程序来找到这条路,要求你告诉他们最少需要花费多少精力

输入描述 Input Description

第一行一个n,接下来n-1行每一行有三个整数u, v, c 。表示节点 u 爬到节点 v 需要花费 c 的精力。

第n+1行有一个整数m表示有m次询问。接下来m行每一行有两个整数 u, v 表示两只虫子所在的节点

输出描述 Output Description

一共有m行,每一行一个整数,表示对于该次询问所得出的最短距离。

样例输入 Sample Input

3
1 0 1
2 0 1
3
1 0
2 0
1 2

样例输出 Sample Output

1
1
2

数据范围及提示 Data Size & Hint

1<=n<=50000,1<=m<=75000,0<=c<=1000

题解和代码

这道题显然是一道LCA的题目,我们可以跑一遍树上倍增。

#include <iostream>
#include <deque>
#include <vector>
#include <cmath>
using namespace std;
int n, m;
struct edge_type
{
int u, v, c, nextu, nextv;
};
edge_type edge[50005];
const int root = 0;
int hu[50005], hv[50005], fa[50005], fac[50005], deep[50005], f[50005][30], maxdeep = 0, fsum[50005][30];
int maxi;
bool vis[50005];
vector<int> son[50005];
deque<int> q;
void bfs_build_tree(const int root)
{
q.push_back(root);
for (int i = 0; i <= n; ++i)
{
vis[i] = false;
}
vis[root] = true;
deep[root] = 1;
int now;
while (!q.empty())
{
now = q.front();
if (maxdeep < deep[now])
maxdeep = deep[now];
q.pop_front();
int i = hu[now];
while (i)
{
if (!vis[edge[i].v])
{
q.push_back(edge[i].v);
vis[edge[i].v] = true;
fa[edge[i].v] = now;
fac[edge[i].v] = edge[i].c;
son[now].push_back(edge[i].v);
deep[edge[i].v] = deep[now] + 1;
}
i = edge[i].nextu;
}
i = hv[now];
while (i)
{
if (!vis[edge[i].u])
{
q.push_back(edge[i].u);
vis[edge[i].u] = true;
fa[edge[i].u] = now;
fac[edge[i].u] = edge[i].c;
son[now].push_back(edge[i].u);
deep[edge[i].u] = deep[now] + 1;
}
i = edge[i].nextv;
}
}
}
void get_f()
{
maxi = log2(double(maxdeep)-0.5);
for (int j = 0; j < n; ++j)
{
f[j][0] = fa[j];
fsum[j][0] = fac[j];
for (int i = 1; i <= 30; ++i)
fsum[j][i] = f[j][i] = -1;
}
fsum[0][0] = f[0][0] = -1;
for (int i = 1; i <= maxi; ++i)
for (int j = 1; j < n; ++j)
if ((1 << i) < deep[j])
{
f[j][i] = f[f[j][i-1]][i-1];
fsum[j][i] = fsum[j][i-1] + fsum[f[j][i-1]][i-1];
}
}
int g(int x, int y)
{
int ret = 0, a = x, b = y;
while (deep[b] > deep[a])
{
for (int i = maxi; i >= 0; --i)
if (deep[f[b][i]] > deep[a])
{
ret += fsum[b][i];
b = f[b][i];
}
if (deep[fa[b]] == deep[a])
{
ret += fac[b];
b = fa[b];
}
}
while (deep[a] > deep[b])
{
for (int i = maxi; i >= 0; --i)
if (deep[f[a][i]] > deep[b])
{
ret += fsum[a][i];
a = f[a][i];
}
if (deep[fa[a]] == deep[b])
{
ret += fac[a];
a = fa[a];
}
}
while (a != b)
{
for (int i = maxi; i >= 0; --i)
if (f[a][i] != f[b][i])
{
ret += (fsum[a][i] + fsum[b][i]);
a = f[a][i];
b = f[b][i];
}
if (fa[a] == fa[b])
{
ret += (fac[a] + fac[b]);
a = fa[a];
b = fa[b];
}
}
return ret;
}
int main()
{
cin >> n;
int u, v, c;
for (int i = 1; i < n; ++i)
{
cin >> u >> v >> c;
edge[i].u = u;
edge[i].v = v;
edge[i].c = c;
edge[i].nextu = hu[u];
edge[i].nextv = hv[v];
hu[u] = i;
hv[v] = i;
}
bfs_build_tree(root);
get_f();
cin >> m;
int a, b;
for (int i = 0; i < m; ++i)
{
cin >> a >> b;
cout << g(a, b) << endl;
}
}

Codevs 2370 小机房的树 LCA 树上倍增的更多相关文章

  1. CodeVs.2370 小机房的树 ( LCA 倍增 最近公共祖先)

    CodeVs.2370 小机房的树 ( LCA 倍增 最近公共祖先) 题意分析 小机房有棵焕狗种的树,树上有N个节点,节点标号为0到N-1,有两只虫子名叫飘狗和大吉狗,分居在两个不同的节点上.有一天, ...

  2. codevs 2370 小机房的树(LCA)

    过了这么长的时间终于开始看LCA了... 有一次训练题卡在LCA当时不会...拖了好久好久...其实现在还是不会... 只会tarjan... 传送门 板子题咯 tarjan的算法就是基于先序遍历的顺 ...

  3. LCA(倍增在线算法) codevs 2370 小机房的树

    codevs 2370 小机房的树 时间限制: 1 s  空间限制: 256000 KB  题目等级 : 钻石 Diamond 题目描述 Description 小机房有棵焕狗种的树,树上有N个节点, ...

  4. Codevs 2370 小机房的树

    2370 小机房的树 时间限制: 1 s 空间限制: 256000 KB 题目等级 : 钻石 Diamond 传送门 题目描述 Description 小机房有棵焕狗种的树,树上有N个节点,节点标号为 ...

  5. codevs——2370 小机房的树

    2370 小机房的树  时间限制: 1 s  空间限制: 256000 KB  题目等级 : 钻石 Diamond 题解       题目描述 Description 小机房有棵焕狗种的树,树上有N个 ...

  6. 放一道比较基础的LCA 的题目把 :CODEVS 2370 小机房的树

    题目描述 Description 小机房有棵焕狗种的树,树上有N个节点,节点标号为0到N-1,有两只虫子名叫飘狗和大吉狗,分居在两个不同的节点上.有一天,他们想爬到一个节点上去搞基,但是作为两只虫子, ...

  7. 【codevs2370】小机房的树 LCA 倍增

    2370 小机房的树  时间限制: 1 s  空间限制: 256000 KB  题目等级 : 钻石 Diamond 题目描述 Description 小机房有棵焕狗种的树,树上有N个节点,节点标号为0 ...

  8. [CodeVS2370] 小机房的树 (LCA, 树链剖分, LCT)

    Description 小机房有棵焕狗种的树,树上有N个节点,节点标号为0到N-1,有两只虫子名叫飘狗和大吉狗,分居在两个不同的节点上.有一天,他们想爬到一个节点上去搞基,但是作为两只虫子,他们不想花 ...

  9. [codevs2370]小机房的树<LCA>

    题目链接:http://codevs.cn/problem/2370/ 这题我还是做了比较久了,因为有人告诉我这是用tarjan离线做 好吧算我是蒟蒻,真心不懂tarjan怎么做,最后还是用倍增做的 ...

随机推荐

  1. angularjs向后台传参,后台收不到数据

    angularjs中封装了一个$http服务,用来请求远程资源 参见:HTTP API 其中封装过的$http.post和$http.get使用起来比较方便 后台是php,用$_POST['name' ...

  2. 调用webservice 总结

    最近做一个项目,由于是在别人框架里开发app,导致了很多限制,其中一个就是不能直接引用webservice . 我们都知道,调用webserivice 最简单的方法就是在 "引用" ...

  3. java 处理XML(dom4j-1.6.1)

    java 处理XML(dom4j-1.6.1) Java 处理xml有很多框架,今天使用主流框架dom4j-1.6.1 下载地址:http://www.dom4j.org/dom4j-1.6.1/ D ...

  4. HTML中图像代替提交按钮

    1. 用图像代替提交按钮 当只有一个提交按钮的时候 ,可以简单的实现,不用添加事件函数,代码是: <input type = "image"' name = ".. ...

  5. web前端学习部落22群分享给需要前端练手项目

    前端学习还是很有趣的,可以较快的上手然后自己开发一些好玩的项目来练手,网上也可以一抓一大把关于前端开发的小项目,可是还是有新手在学习的时候不知道可以做什么,以及怎么做,因此,就整理了一些前端项目教程, ...

  6. 统一配置管理-百度disconf

    之前一直采用properties文件管理配置信息,若是集群则每个机器上都要拷贝一份,每次修改也需要依次修改.一直在寻找统一修改,实时生效,方便修改,分环境分系统的配置管理,自己也在整理设计,若找不到合 ...

  7. em换算px,rem换算px

    <body> <style type="text/css" rel="stylesheet"> html,body{ font-size ...

  8. HDU 3642 Get The Treasury (线段树扫描线)

    题意:给你一些长方体,问你覆盖三次及以上的体积有多大 首先我们观察x轴y轴一样很大,但是z轴很小,所以我们可以枚举z轴(-500,500),注意我们枚举的是每一段长度为一的z轴的xy轴的面积而不是点. ...

  9. 使用lnmp一键安装包后yum源出现的问题与解决

    遇到一个问题就是执行 Yum update 或使用yum安装软件等命令的时候会有一个链接报404,使用find查找到该链接与kbsingh-CentOS-Extras.repo这个包有关,删除后yum ...

  10. mysql导入导出,及错误记录

    进入mysql的bin目录,如果mysql的bin添加了环境变量则不用. 导出,不指定编码则默认为:utf8mb4.: mysqldump -u root -h 127.0.0.1 -P 3307 - ...