【LG3248】[HNOI2016]树

题面

洛谷

题解

因为每次你加入的点是原树上某一棵子树

那么我们一次加入一个点,代表一棵子树加到大树下面

那么我们要找到一个点在一个大点中用主席树在\(dfs\)序中\(kth\)即可

询问的话,先将所有的点权(深度)转化为边权

查询时先将两点跳到它所在大点的根

再倍增跳到大点1(原树)的下面,再在原树上倍增跳一跳即可。

虽然说起来容易,但是其实还是很码的qaq。

代码

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
using namespace std; template<typename T>
inline void read(T &data) {
data = 0; T w = 1;
char ch = getchar();
while (ch != '-' && (!isdigit(ch))) ch = getchar();
if (ch == '-') w = -1, ch = getchar();
while (isdigit(ch)) data = data * 10 + (ch ^ 48), ch = getchar();
data *= w;
}
typedef long long ll;
const int MAX_N = 1e5 + 5;
int N, M, Q, lg[MAX_N];
namespace Ptree {
struct Node { int ls, rs, v; } t[MAX_N * 20];
int root[MAX_N], tot;
void build(int &o, int l = 1, int r = N) {
o = ++tot;
if (l == r) return ;
int mid = (l + r) >> 1;
build(t[o].ls, l, mid);
build(t[o].rs, mid + 1, r);
}
void insert(int &o, int p, int v, int l = 1, int r = N) {
o = ++tot, t[o] = t[p];
t[o].v++;
if (l == r) return ;
int mid = (l + r) >> 1;
if (v <= mid) insert(t[o].ls, t[p].ls, v, l, mid);
else insert(t[o].rs, t[p].rs, v, mid + 1, r);
}
int query(int x, int y, int k, int l = 1, int r = N) {
if (l == r) return l;
int mid = (l + r) >> 1, s = t[t[y].ls].v - t[t[x].ls].v;
if (k <= s) return query(t[x].ls, t[y].ls, k, l, mid);
else return query(t[x].rs, t[y].rs, k - s, mid + 1, r);
}
}
namespace Template {
int fa[20][MAX_N], dep[MAX_N], pos[MAX_N], R[MAX_N], cnt, L[MAX_N];
struct Graph { int next, to; } e[MAX_N << 1]; int fir[MAX_N], e_cnt;
void clearGraph() { memset(fir, -1, sizeof(fir)); e_cnt = 0; }
void Add_Edge(int u, int v) { e[e_cnt] = (Graph){fir[u], v}; fir[u] = e_cnt++; }
void dfs(int x, int f) {
L[pos[x] = ++cnt] = x;
fa[0][x] = f;
dep[x] = dep[f] + 1;
for (int i = 1; i < 20; i++) fa[i][x] = fa[i - 1][fa[i - 1][x]];
for (int i = fir[x]; ~i; i = e[i].next) if (e[i].to != f) dfs(e[i].to, x);
R[x] = cnt;
} int Dis(int x, int y) {
if (dep[x] < dep[y]) swap(x, y);
int res = dep[x] - dep[y];
for (int i = 20 - 1; ~i; i--)
if (dep[fa[i][x]] >= dep[y]) x = fa[i][x];
if (x == y) return res;
for (int i = 20 - 1; ~i; i--)
if (fa[i][x] != fa[i][y])
res += 1 << (i + 1), x = fa[i][x], y = fa[i][y];
return res + 2;
}
} namespace BigTree {
int N, fa[20][MAX_N], dep[MAX_N], pre[MAX_N];
ll dis[20][MAX_N], pos[MAX_N], R[MAX_N], cur, link[MAX_N];
int getRoot(ll x) {
int l = 1, r = N;
while (l <= r) {
int mid = (l + r) >> 1;
if (pos[mid] <= x) l = mid + 1;
else r = mid - 1;
}
return r;
} int getPre(ll x) {
int rt = getRoot(x);
return Ptree::query(Ptree::root[Template::pos[pre[rt]] - 1], Ptree::root[Template::R[pre[rt]]],
x - pos[rt] + 1);
} void Init() {
pos[1] = N = dep[1] = pre[1] = 1;
cur = R[1] = ::N;
int x; ll to;
for (int i = 1; i <= M; i++) {
read(x), read(to);
int rt = getRoot(to);
++N, dep[N] = dep[rt] + 1, link[N] = to, pre[N] = x;
pos[N] = cur + 1, R[N] = cur + Template::R[x] - Template::pos[x] + 1;
cur = R[N], fa[0][N] = rt;
dis[0][N] = Template::dep[getPre(to)] - Template::dep[pre[rt]] + 1;
for (int j = 1; j < 20; j++)
fa[j][N] = fa[j - 1][fa[j - 1][N]], dis[j][N] = dis[j - 1][N] + dis[j - 1][fa[j - 1][N]];
}
} ll Dis(ll x, ll y) {
ll ans = 0;
int fx = getRoot(x), fy = getRoot(y);
if (fx == fy) return Template::Dis(getPre(x), getPre(y));
if (dep[fx] < dep[fy]) swap(x, y), swap(fx, fy);
ans += Template::dep[getPre(x)] - Template::dep[pre[fx]], x = fx;
for (int i = 20 - 1; ~i; i--)
if (dep[fa[i][x]] > dep[fy]) ans += dis[i][x], x = fa[i][x];
if (getRoot(link[x]) == fy)
return ans + 1 + Template::Dis(getPre(link[x]), getPre(y));
ans += Template::dep[getPre(y)] - Template::dep[pre[fy]], y = fy;
if (dep[x] > dep[y]) ans += dis[0][x], x = fa[0][x];
for (int i = 20 - 1; ~i; i--)
if (fa[i][x] != fa[i][y]) ans += dis[i][x] + dis[i][y], x = fa[i][x], y = fa[i][y];
x = link[x], y = link[y];
ans += 2;
return ans + Template::Dis(getPre(x), getPre(y));
}
} int main() {
#ifndef ONLINE_JUDGE
freopen("cpp.in", "r", stdin);
#endif
read(N), read(M), read(Q);
for (int i = 2; i <= N; i++) lg[i] = lg[i >> 1] + 1;
Template::clearGraph();
for (int i = 1, a, b; i < N; i++)
read(a), read(b), Template::Add_Edge(a, b), Template::Add_Edge(b, a);
Template::dfs(1, 0);
Ptree::build(Ptree::root[0]);
for (int i = 1; i <= N; i++) Ptree::insert(Ptree::root[i], Ptree::root[i - 1], Template::L[i]);
BigTree::Init();
ll x, y;
while (Q--) read(x), read(y), printf("%lld\n", BigTree::Dis(x, y));
return 0;
}

【LG3248】[HNOI2016]树的更多相关文章

  1. BZOJ 4539: [Hnoi2016]树 [主席树 lca]

    4539: [Hnoi2016]树 题意:不想写.复制模板树的子树,查询两点间距离. *** 终于有一道会做的题了...... 画一画发现可以把每次复制的子树看成一个大点来建一棵树,两点的lca一定在 ...

  2. 4539: [Hnoi2016]树

    4539: [Hnoi2016]树 链接 分析: 主席树+倍增. 代码: #include<cstdio> #include<algorithm> #include<cs ...

  3. [BZOJ4539][HNOI2016]树(主席树)

    4539: [Hnoi2016]树 Time Limit: 40 Sec  Memory Limit: 256 MBSubmit: 746  Solved: 292[Submit][Status][D ...

  4. [HNOI2016]树(可持久化线段树+树上倍增)

    [HNOI2016]树(可持久化线段树+树上倍增) 题面 给出一棵n个点的模板树和大树,根为1,初始的时候大树和模板树相同.接下来操作m次,每次从模板树里取出一棵子树,把它作为新树里节点y的儿子.操作 ...

  5. BZOJ4539: [Hnoi2016]树

    复制的树缩点,主席树查k小,毫无技术含量,纯码农题. #include<bits/stdc++.h> #define u first #define v second #define F ...

  6. bzoj 4539: [Hnoi2016]树

    Description 小A想做一棵很大的树,但是他手上的材料有限,只好用点小技巧了.开始,小A只有一棵结点数为N的树,结 点的编号为1,2,-,N,其中结点1为根:我们称这颗树为模板树.小A决定通过 ...

  7. [HNOI2016]树

    Description 小A想做一棵很大的树,但是他手上的材料有限,只好用点小技巧了.开始,小A只有一棵结点数为N的树,结 点的编号为1,2,…,N,其中结点1为根:我们称这颗树为模板树.小A决定通过 ...

  8. 2019.03.25 bzoj4539: [Hnoi2016]树(主席树+倍增)

    传送门 题意:给一棵大树,令一棵模板树与这棵树相同,然后进行mmm次操作,每次选择模板树中的一个节点aaa和大树中一个节点bbb,把aaa这棵子树接在bbb上面,节点编号顺序跟aaa中的编号顺序相同. ...

  9. 洛谷P3248 [HNOI2016]树(主席树 倍增 )

    题意 题目链接 Sol 从上午九点淦到现在qwq 思路比较简单,就是把每次加入的一坨点看成一个,然后直接倍增搞.. 然后慢慢调就可以了... 最后数量级会到达\(10^{10}\),所以应该开long ...

随机推荐

  1. [C# | XML] XML 反序列化解析错误:<xml xmlns=''> was not expected. 附通用XML到类解析方法

    使用 XML 反化时出现错误: public static TResult GetObjectFromXml<TResult>(string xmlString) { TResult re ...

  2. 使用Github的高级搜索功能

    使用Github的高级搜索功能 1. 首先,提供Github高级搜索帮助页面 https://help.github.com/categories/search/ 2. 搜索语法 https://he ...

  3. Linux wc命令详解

    wc常见命令参数 wc -l : 统计行 wc -c: 统计字节数 wc -m:统计字符数,不能与-c同时使用 wc -w:统计字数 wc -L:打印最长长度 注意: wc 可以直接后面跟文件使用,但 ...

  4. Springboot 报application.properites文件找不到的解决方法

        部署项目遇到了找不到application.properties的问题.网上搜了找不到答案,后面经过测试发现,问题在于clean了maven之后,target包还没删除,所以编译的时候才会出现 ...

  5. Spring+微信小程序 卡券打通

    近期公司项目需要使用到微信卡券模块,主要做的是在小程序打通微信卡券,实现小程序领取卡券的功能效果. 简单说下涉及的东西: Springboot—使用springboot做后端接口,非常便捷 并且根本是 ...

  6. EclEmma安装与使用

    安装 EclEmma 插件的过程和大部分 Eclipse 插件相同,我们既可以通过 Eclipse 标准的 Update 机制来远程安装 EclEmma 插件(图 1),也可以从站点(参阅参考资源)下 ...

  7. centos安装不上的问题

    Installing VMware Tools, please wait...mount: special device /dev/hda does not existmount: block dev ...

  8. 1055. [HAOI2008]玩具取名【区间DP】

    Description 某人有一套玩具,并想法给玩具命名.首先他选择WING四个字母中的任意一个字母作为玩具的基本名字.然后 他会根据自己的喜好,将名字中任意一个字母用“WING”中任意两个字母代替, ...

  9. 【洛谷】【动态规划/二维背包】P1855 榨取kkksc03

    [题目描述:] ... (宣传luogu2的内容被自动省略) 洛谷的运营组决定,如果...,那么他可以浪费掉kkksc03的一些时间的同时消耗掉kkksc03的一些金钱以满足自己的一个愿望. Kkks ...

  10. net mvc中实现记录用户登录信息(记住登录效果)

    现记录用户登录信息(记住登录效果) 本文讲述了使用cookies实现网站记住登录效果,效果如下: 主要实现方法,当用户选择记住登录时建立cookies保存用户名和用户密码,当用户登录不选择记住登录时, ...