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 ...
随机推荐
- 英语单词delimiter
delimiter 来源——命令参数 [root@centos7 ~]# cat /good/passwd root:x:::root:/root:/bin/bash bin:x:::bin:/bin ...
- linux查找一个文件的路径
- Maven开始
1:加入Maven插件: 这句话的意思是: 从本地仓库找到相应的jar包 <localRepository>F:\RepMaven</localRepository 2:创建一个Ma ...
- 【LeetCode 73】矩阵置零
题目链接 [题解] 如果a[i][j]==0. 就把第i行的第一个数字置为0 然后把第j列的第一个数字置为0 最后再处理下每行第一个为0的行.每列第一个为0的列. (第一行和第一列都得用同一个位置处理 ...
- vue项目git
https://github.com/renrenio/renren-fast-vue https://github.com/hzlshen/vue-project
- C#后台获取post参数
public static string GetQueryString(string key) { if (HttpContext.Current.Request[key] == null) retu ...
- Ant Design Pro (中后台系统)教程
一.概念:https://pro.ant.design/docs/getting-started-cn(官方网站) 1.Ant Design Pro 是什么: https://www.cnblogs ...
- sonarqube6.7.1使用
1.插件安装 方法1.登入sonarqube-web安装 admin/admin 配置--应用市场--全部 英文片:administration--configuration--marketplace ...
- 2018-2019-2 20175223 实验四 《Android开发基础》实验报告
目录 北京电子科技学院(BESTI)实验报告 实验名称:实验四 Android程序设计 实验内容.步骤与体会: 一.实验四 Android程序设计-1 二.实验四 Android程序设计-2 三.实验 ...
- c++文件拷贝
#include<fstream>void Copyfile( char* FileSource, char* FileItem ){fstream fsCopee( FileSource ...