bzoj 3626
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的更多相关文章
- [BZOJ 3626] [LNOI2014] LCA 【树链剖分 + 离线 + 差分询问】
题目链接: BZOJ - 3626 题目分析 考虑这样的等价问题,如果我们把一个点 x 到 Root 的路径上每个点的权值赋为 1 ,其余点的权值为 0,那么从 LCA(x, y) 的 Depth 就 ...
- BZOJ 3626: [LNOI2014]LCA [树链剖分 离线|主席树]
3626: [LNOI2014]LCA Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 2050 Solved: 817[Submit][Status ...
- bzoj 3626 [LNOI2014]LCA(离线处理+树链剖分,线段树)
3626: [LNOI2014]LCA Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1272 Solved: 451[Submit][Status ...
- bzoj 3626: [LNOI2014]LCA 离线+树链剖分
3626: [LNOI2014]LCA Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 426 Solved: 124[Submit][Status] ...
- BZOJ 3626: [LNOI2014]LCA( 树链剖分 + 离线 )
说多了都是泪啊...调了这么久.. 离线可以搞 , 树链剖分就OK了... -------------------------------------------------------------- ...
- BZOJ 3626: [LNOI2014]LCA 树链剖分 线段树 离线
http://www.lydsy.com/JudgeOnline/problem.php?id=3626 LNOI的树链剖分题没有HAOI那么水,学到的东西还是很多的. 我如果现场写,很难想出来这种题 ...
- AC日记——[LNOI2014]LCA bzoj 3626
3626 思路: 离线操作+树剖: 代码: #include <bits/stdc++.h> using namespace std; #define maxn 100005 #defin ...
- BZOJ 3626 [LNOI2014]LCA:树剖 + 差分 + 离线【将深度转化成点权之和】
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=3626 题意: 给出一个n个节点的有根树(编号为0到n-1,根节点为0,n <= 50 ...
- BZOJ 3626 [LNOI2014]LCA 树剖+(离线+线段树 // 在线+主席树)
BZOJ 4012 [HNOI2015]开店 的弱化版,离线了,而且没有边权(长度). 两种做法 1 树剖+离线+线段树 这道题求的是一个点zzz与[l,r][l,r][l,r]内所有点的lcalca ...
随机推荐
- Spring学习总结(2)- AOP
一,什么是AOP AOP(Aspect Oriented Programming)意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术.AOP是OOP的延续,是软件开发中 ...
- struts2---访问WEB
一:在Action中,可以通过以下方式访问WEB的HttpSession,HttpServletRequest,HttpServletResponse等资源 与Servlet API解耦的访问方式 通 ...
- APIO2019 练习赛 Wedding cake——思路+高精度
题目大意: 给 n ( n<=1e5 ) 个数 \( a_i \) (\( a_i \) <=1e5),需要构造 n 个实数使得它们的和是 1 ,并且第 i 个实数必须小数点后恰好有 \( ...
- 在使用KVO遇到的一个问题
在项目开发中定义了一个单例对象RHUserData的对象,RHOLUserInfo类是单例对象的一个property属性,RHOLUserInfo里面有个userId的属性,在其他类里面进行设置KVO ...
- angualr项目引入容联 七陌7mroo
最近项目要求在注册页面增加客服服务浮窗,各种查找资料准备采用7moor来实现.现记录一下实现过程,便于后期查看: 引入7moor浮窗有两种方式: 1.h5方式,这种情况一般是单独打开新页面即可: 直接 ...
- element table 通过selection-change选中的索引删除
<el-table :row-class-name="tableRowClassName" @selection-change="handleSelectionCh ...
- Gym-100676E Time Limit Exceeded?
原题链接 https://odzkskevi.qnssl.com/1110bec98ca57b5ce6aec79b210d2849?v=1491063604 ********************* ...
- 高并发大流量专题---11、Web服务器的负载均衡
高并发大流量专题---11.Web服务器的负载均衡 一.总结 一句话总结: 推荐使用nginx七层(应用层)负载均衡的实现:配置那是相当的简单 http{ upstream cluster{ serv ...
- 6105 - deauth after EAPOL key exchange sequence
wifi无法连接公司的网络 Warning Error in Event Log - deauth after EAPOL key exchange sequence https://forums.i ...
- css控制显示超出多少行以后开始出现省略号的写法
display: -webkit-box; display: -moz-box; text-overflow: -o-ellipsis-lastline; text-overflow: ellipsi ...