http://www.lydsy.com/JudgeOnline/problem.php?id=3626

让我比较惊讶的一道链剖裸题(' '    ) 做法很精妙

首先我们考虑对于单个询问时可以拆分成(1, l - 1, z) 和 (1, r, z) 的, 然后考虑对于每一次询问可以表示为将(1, l) 的所有点到根的全部加1 然后求z到根路径的的和。 所以将询问离线, 按询问的l值排序,每一次遇到新的l值就将这一段的点到根的路径全部加1,然后查询即可

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std; typedef long long ll;
const ll maxn = 100001;
const ll mod = 201314; ll n, m; ll int_get() {
ll x = 0; char c = (char)getchar(); bool f = 0;
while(!isdigit(c)) {
if(c == '-') f = 1;
c = (char)getchar();
}
while(isdigit(c)) {
x = x * 10 + (int)(c - '0');
c = (char)getchar();
}
if(f) x = -x;
return x;
} struct seg {
ll data, lazy;
seg *l, *r;
}tr[maxn * 3];ll sege = 0;
seg* root; void test(seg* x, ll l, ll r) {
cout << l <<" "<< r <<" "<< x-> data <<" "<< x-> lazy << endl;
if(l ^ r) {
ll mid = (l + r) >> 1;
test(x-> l, l, mid), test(x-> r, mid + 1, r);
}
} seg* build(ll l, ll r) {
seg* x = tr + sege ++;
if(l ^ r) {
ll mid = (l + r) >> 1;
x-> l = build(l, mid);
x-> r = build(mid + 1, r);
}
return x;
} void update(seg* x) {
if(x-> l) x-> data = x-> l-> data + x-> r-> data;
} void pushdown(seg* x, ll l, ll r) {
if(x-> l && x-> lazy != 0) {
ll mid = (l + r) >> 1;
x-> l-> lazy += x-> lazy, x-> l-> data += (mid - l + 1) * x-> lazy;
x-> r-> lazy += x-> lazy, x-> r-> data += (r - mid) * x-> lazy;
x-> lazy = 0;
}
} void addlazy(seg* x, ll l, ll r, ll ls, ll rs, ll v) {
if(l == ls && r == rs) x-> data += v * (rs - ls + 1), x-> lazy += v;
else {
pushdown(x, l, r);
ll mid = (l + r) >> 1;
if(rs <= mid) addlazy(x-> l, l, mid, ls, rs, v);
else if(ls > mid) addlazy(x-> r, mid + 1, r, ls, rs, v);
else addlazy(x-> l, l, mid, ls, mid, v), addlazy(x-> r, mid + 1, r, mid + 1, rs, v);
update(x);
}
} ll ask(seg* x, ll l, ll r, ll ls, ll rs) {
if(l == ls && r == rs) return x-> data;
else {
pushdown(x, l, r);
ll mid = (l + r) >> 1;
if(rs <= mid) return ask(x-> l, l, mid, ls, rs);
else if(ls > mid) return ask(x-> r, mid + 1, r, ls, rs);
else return ask(x-> l, l, mid, ls, mid) + ask(x-> r, mid + 1, r, mid + 1, rs);
}
} struct edge {
ll t;
edge* next;
}e[maxn * 2], *head[maxn]; ll ne = 0; void addedge(ll f, ll t) {
e[ne].t = t, e[ne].next = head[f], head[f] = e + ne ++;
} ll h[maxn], size[maxn], fa[maxn], un[maxn], map[maxn], num = 0; void dfs(ll x, ll pre) {
fa[x] = pre, size[x] = 1; h[x] = h[pre] + 1;
for(edge* p = head[x]; p; p = p-> next) {
if(p-> t != pre) dfs(p-> t, x), size[x] += size[p-> t];
}
} void divide(ll x, ll Un) {
un[x] = Un, map[x] = ++ num;
if(size[x] == 1) return ;
ll Max = 0, pos;
for(edge* p = head[x]; p; p = p-> next) {
if(p-> t != fa[x] && size[p-> t] > Max) Max = size[p-> t], pos = p-> t;
}
divide(pos, Un);
for(edge* p = head[x]; p; p = p-> next) {
if(p-> t != fa[x] && p-> t != pos) divide(p-> t, p-> t);
}
} void add(ll a, ll b, ll v) {
ll ls, rs;
while(un[a] != un[b]) {
if(h[un[a]] > h[un[b]]) {
ls = map[un[a]], rs = map[a];
addlazy(root, 1, n, ls, rs, v);
a = fa[un[a]];
}
else {
ls = map[un[b]], rs = map[b];
addlazy(root, 1, n, ls, rs, v);
b = fa[un[b]];
}
}
ls = map[a], rs = map[b];
if(ls > rs) swap(ls, rs);
addlazy(root, 1, n, ls, rs, v);
} ll get(ll a, ll b) {
ll ret = 0;
ll ls, rs;
while(un[a] != un[b]) {
if(h[un[a]] > h[un[b]]) {
ls = map[un[a]], rs = map[a];
ret += ask(root, 1, n, ls, rs);
a = fa[un[a]];
}
else {
ls = map[un[b]], rs = map[b];
ret += ask(root, 1, n, ls, rs);
b = fa[un[b]];
}
}
ls = map[a], rs = map[b];
if(ls > rs) swap(ls, rs);
ret += ask(root, 1, n, ls, rs);
return ret;
} struct block {
ll x, p, pos, fl;
}o[maxn]; ll oe = 0; bool cmp(block a, block b) {
return a. x < b. x;
} ll ans[maxn]; void read() {
n = int_get(); m = int_get();
for(ll i = 2; i <= n; ++ i) {
ll u = int_get() + 1;
addedge(u, i), addedge(i, u);
}
for(ll i = 1; i <= m; ++ i) {
ll a, b, c;
a = int_get() + 1, b = int_get() + 1, c = int_get() + 1;
++ oe, o[oe]. x = a - 1, o[oe]. p = c, o[oe]. pos = i, o[oe]. fl = -1;
++ oe, o[oe]. x = b, o[oe]. p = c, o[oe].pos = i, o[oe].fl = 1;
}
dfs(1, 0), divide(1, 1);
root = build(1, n);
} void sov() {
sort(o + 1, o + 1 + oe, cmp); ll pl = 0;
for(ll i = 1; i <= oe; ++ i) {
while(pl < o[i]. x) {
++ pl; add(1, pl, 1);
}
//test(root, 1, n); cout << endl;
ans[o[i]. pos] += o[i]. fl * get(1, o[i]. p);
}
for(ll i = 1; i <= m; ++ i) printf("%lld\n", (ans[i] % mod + mod)% mod);
} int main() {
//freopen("test.in", "r", stdin);
//freopen("test.out", "w", stdout);
read();
sov();
return 0;
}

bzoj 3626的更多相关文章

  1. [BZOJ 3626] [LNOI2014] LCA 【树链剖分 + 离线 + 差分询问】

    题目链接: BZOJ - 3626 题目分析 考虑这样的等价问题,如果我们把一个点 x 到 Root 的路径上每个点的权值赋为 1 ,其余点的权值为 0,那么从 LCA(x, y) 的 Depth 就 ...

  2. BZOJ 3626: [LNOI2014]LCA [树链剖分 离线|主席树]

    3626: [LNOI2014]LCA Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 2050  Solved: 817[Submit][Status ...

  3. bzoj 3626 [LNOI2014]LCA(离线处理+树链剖分,线段树)

    3626: [LNOI2014]LCA Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1272  Solved: 451[Submit][Status ...

  4. bzoj 3626: [LNOI2014]LCA 离线+树链剖分

    3626: [LNOI2014]LCA Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 426  Solved: 124[Submit][Status] ...

  5. BZOJ 3626: [LNOI2014]LCA( 树链剖分 + 离线 )

    说多了都是泪啊...调了这么久.. 离线可以搞 , 树链剖分就OK了... -------------------------------------------------------------- ...

  6. BZOJ 3626: [LNOI2014]LCA 树链剖分 线段树 离线

    http://www.lydsy.com/JudgeOnline/problem.php?id=3626 LNOI的树链剖分题没有HAOI那么水,学到的东西还是很多的. 我如果现场写,很难想出来这种题 ...

  7. AC日记——[LNOI2014]LCA bzoj 3626

    3626 思路: 离线操作+树剖: 代码: #include <bits/stdc++.h> using namespace std; #define maxn 100005 #defin ...

  8. BZOJ 3626 [LNOI2014]LCA:树剖 + 差分 + 离线【将深度转化成点权之和】

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=3626 题意: 给出一个n个节点的有根树(编号为0到n-1,根节点为0,n <= 50 ...

  9. BZOJ 3626 [LNOI2014]LCA 树剖+(离线+线段树 // 在线+主席树)

    BZOJ 4012 [HNOI2015]开店 的弱化版,离线了,而且没有边权(长度). 两种做法 1 树剖+离线+线段树 这道题求的是一个点zzz与[l,r][l,r][l,r]内所有点的lcalca ...

随机推荐

  1. SSH小应用

    1:Spring整合Hibernate <?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE hi ...

  2. HTML基础知识笔记摘要

    HTML互联网三大基石:1.HTML:显示数据2.HTTP:传输数据 http传输协议3.URL:定位数据协议://ip地址或主机名:端口/网络中的内容... HTML(hyper text make ...

  3. visual studio code的使用

    1.添加代码片段 参考:https://blog.csdn.net/qq_36370731/article/details/83014439 2.在vscode上运行Git 先打开vscode内置的G ...

  4. Codechef BINOMSUM

    题意:(复制sunset的)有\(T\)天,每天有\(K\)个小时,第\(i\)天有\(D+i−1\)道菜,第一个小时你选择\(L\)道菜吃,接下来每个小时你可以选择吃一道菜或者选择\(A\)个活动中 ...

  5. CodeForces - 1051D (线性DP)

    题目:https://codeforces.com/problemset/problem/1051/D 题意:一个2行n列的矩形,上面有黑白块,然后问你怎么布置才能有k个连通块,问有多少种方案数 思路 ...

  6. TCP/IP协议 和 如何实现 互联网上点对点的通信

    1.参考:https://www.cnblogs.com/onepixel/p/7092302.html   TCP/IP 协议采用4层结构,分别是应用层.传输层.网络层 和 链路层   http 属 ...

  7. docstoc对Scribd的威胁比SlideShare还要大。

    docstoc,这是在TechCrunch40互联网交流会上崭露头角的个性化服务.docstoc为用户提供了在线存储.分享以及交流文档的互联网服务.与Scribd相同的是,除了文档分享功能以外,doc ...

  8. xcodebuild 自动化打包

    altool 文档 使用xcode自带的xcodebuild 命令通过脚本进行打包 打包->导出ipa, 两行关键的脚本代码 1.Archive xcodebuild archive -arch ...

  9. git使用记录九:开发中临时加塞了紧急任务怎么处理

    开发中临时加塞了紧急任务怎么处理 隐藏工作区域 git stash git status 查询隐藏的列表 git stash list 处理完bug,提交之后,再恢复隐藏的工作区域 git stash ...

  10. Linux安装Apache报错:Cannot find a valid baseurl for repo: base/7/x86_64解决方案

    最近使用CentOS7学习,安装安装Apache时候,使用yum安装Apache报错:本文适合CentOS7和RHEL7 # yum install httpd 出现:cannot find a va ...