标签:str class 情况 接下来 动态 使用 区间查询 name 不能
题目描述
输入
输出
对于每个事件1, 输出询问的果子数.
样例输入
5
1 2
2 3
2 4
1 5
3
0 1 1
0 2 3
1 2 3 1 1 4
样例输出
13
题解
树链剖分+线段树
一眼看上去树剖线段树傻*题,然而实际看上去却不是这样。
由于每次选择了某些链,它们重叠的部分是不能计算的,所以情况比较复杂。
我们可以换一个角度去思考,把“计算链的点权和”改为“标记链上的点,并计算被标记的点的点权和”。
这有什么用?
由于区间的标记仅为0/1,所以对某段区间打标记时可以用线段树维护和,并不需要加减性质。
于是有:将每条链对应的每段区间在线段树中标记上,然后统计线段树中所有被标记的点的权值和,最后删除所有点的标记。
树剖+线段树区间修改区间查询,时间复杂度$O(Qk\log ^2n)$。
需要稍微注意一下双标记的处理,如果有标记的标记(这话这么别扭= = 选择标记)则应当立刻优先处理。
至于mod maxlongint,可以使用int自然溢出,最后结果和maxlongint取与。
至今不知道“所有链随机生成且所有链上的点的最近公共祖先为链的端点”这个条件有什么用= =
#include <cstdio> #include <cstring> #include <algorithm> #define N 200010 #define lson l , mid , x << 1 #define rson mid + 1 , r , x << 1 | 1 using namespace std; int head[N] , to[N << 1] , next[N << 1] , cnt; int fa[N] , deep[N] , si[N] , bl[N] , pos[N] , last[N] , tot; int sum[N << 2] , val[N << 2] , add[N << 2] , tag[N << 2] , n; void ins(int x , int y) { to[++cnt] = y , next[cnt] = head[x] , head[x] = cnt; } void dfs1(int x) { int i; si[x] = 1; for(i = head[x] ; i ; i = next[i]) if(to[i] != fa[x]) fa[to[i]] = x , deep[to[i]] = deep[x] + 1 , dfs1(to[i]) , si[x] += si[to[i]]; } void dfs2(int x , int c) { int i , k = 0; bl[x] = c , pos[x] = ++tot; for(i = head[x] ; i ; i = next[i]) if(to[i] != fa[x] && si[to[i]] > si[k]) k = to[i]; if(k) { dfs2(k , c); for(i = head[x] ; i ; i = next[i]) if(to[i] != fa[x] && to[i] != k) dfs2(to[i] , to[i]); } last[x] = tot; } void pushup(int x) { sum[x] = sum[x << 1] + sum[x << 1 | 1] , val[x] = val[x << 1] + val[x << 1 | 1]; } void pushdown(int l , int r , int x) { if(add[x]) { int mid = (l + r) >> 1; sum[x << 1] += add[x] * (mid - l + 1) , sum[x << 1 | 1] += add[x] * (r - mid); add[x << 1] += add[x] , add[x << 1 | 1] += add[x] , add[x] = 0; } if(~tag[x]) { val[x << 1] = sum[x << 1] * tag[x] , val[x << 1 | 1] = sum[x << 1 | 1] * tag[x]; tag[x << 1] = tag[x << 1 | 1] = tag[x] , tag[x] = -1; } } void update(int b , int e , int a , int l , int r , int x) { if(b <= l && r <= e) { sum[x] += a * (r - l + 1) , add[x] += a; return; } pushdown(l , r , x); int mid = (l + r) >> 1; if(b <= mid) update(b , e , a , lson); if(e > mid) update(b , e , a , rson); pushup(x); } void color(int b , int e , int c , int l , int r , int x) { if(b <= l && r <= e) { val[x] = c * sum[x] , tag[x] = c; return; } pushdown(l , r , x); int mid = (l + r) >> 1; if(b <= mid) color(b , e , c , lson); if(e > mid) color(b , e , c , rson); pushup(x); } void solve(int x , int y) { while(bl[x] != bl[y]) { if(deep[bl[x]] < deep[bl[y]]) swap(x , y); color(pos[bl[x]] , pos[x] , 1 , 1 , n , 1) , x = fa[bl[x]]; } if(deep[x] > deep[y]) swap(x , y); color(pos[x] , pos[y] , 1 , 1 , n , 1); } int main() { int m , i , opt , k , x , y; scanf("%d" , &n); for(i = 1 ; i < n ; i ++ ) scanf("%d%d" , &x , &y) , ins(x , y) , ins(y , x); dfs1(1) , dfs2(1 , 1); memset(tag , -1 , sizeof(tag)); scanf("%d" , &m); while(m -- ) { scanf("%d" , &opt); if(opt == 0) scanf("%d%d" , &x , &y) , update(pos[x] , last[x] , y , 1 , n ,1); else { scanf("%d" , &k); while(k -- ) scanf("%d%d" , &x , &y) , solve(x , y); printf("%d\n" , val[1] & 0x7fffffff); color(1 , n , 0 , 1 , n , 1); } } return 0; }
标签:str class 情况 接下来 动态 使用 区间查询 name 不能
原文地址:http://www.cnblogs.com/GXZlegend/p/7113001.html