题意:一个树,支持两种操作:1.将深度为L的节点权置加上X;2.求以x为根节点的子树上节点权置之和.根节点深度为0.

分析:考虑用树状数组维护节点权置,按dfs序下标查询.记录每个深度节点的个数.如果每次都暴力维护子树上每一层的节点,则会超时.

要用分块来解决.对于节点数量小于\(\sqrt{N}\)的层数,用树状数组维护;否则将该层记录,修改时单独记录.

查询时,答案分成两份:树状数组中维护的子树区间的和;以及属于根为所查节点x子树,且层号是超过\(\sqrt{N}\)的节点的权置和.后者查询时,对每一个超过\(\sqrt{N}\)的层号,求其层中节点在x子树区间中的节点权置之和.

#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
const int MAXN = 2e5+5;
struct Edge{
int v,next;
}edges[MAXN<<1];
int head[MAXN], tot,N,deep,dfs_cnt;
int L[MAXN],R[MAXN];
LL bit[MAXN<<1];
vector<int> dp[MAXN];
vector<int> bg; void init(){
memset(bit,0,sizeof(bit));
memset(head,-1,sizeof(head));
tot =0;
deep = dfs_cnt = 0;
} void AddEdge(int u,int v){
edges[tot] = (Edge){v,head[u]};
head[u] = tot++;
} void add(int pos,LL val){
for(int i=pos; i<=N; i+= (i&-i)) bit[i] += val;
} LL sum(int pos)
{
LL res=0;
for(int i=pos; i ;i-= (i&-i)) res+= bit[i];
return res;
} void dfs(int u,int fa,int d)
{
L[u] = ++ dfs_cnt;
deep = max(deep,d);
dp[d].push_back(L[u]);
for(int i=head[u];~i;i=edges[i].next){
int v = edges[i].v;
if(v!=fa) dfs(v,u,d+1);
}
R[u] = dfs_cnt;
}
LL ans[MAXN]; int main()
{
#ifndef ONLINE_JUDGE
freopen("in.txt","r",stdin);
freopen("out.txt","w",stdout);
#endif
int Q,op,u,v,tmp;
scanf("%d %d",&N, &Q);
init();
memset(ans,0,sizeof(ans));
bg.clear();
for(int i=0;i<=N;++i) dp[i].clear(); for(int i=1;i<=N-1;++i){
scanf("%d %d",&u,&v);
AddEdge(u,v), AddEdge(v,u);
}
dfs(1,-1,0);
int block = sqrt(1.0*N);
for(int i=0;i<=deep;++i){ //该层节点大于sqrt(n)的单独保存
if(dp[i].size()>block) bg.push_back(i);
}
while(Q--){
scanf("%d",&op);
if(op==1){
int lev; LL x;
scanf("%d %lld",&lev, &x);
if(dp[lev].size()>block){ //节点数大的打上标记
ans[lev] += x;
}
else{ //该层节点数小的直接暴力更新.
int siz = dp[lev].size();
for(int i=0;i<siz;++i){
add(dp[lev][i],x);
}
}
}
else{
scanf("%d",&u);
LL res = sum(R[u]) - sum(L[u]-1); //先求层数小的节点和
int siz = bg.size();
for(int i=0;i<siz;++i){
int lev = bg[i];
res += (upper_bound(dp[lev].begin(),dp[lev].end(),R[u])-
lower_bound(dp[lev].begin(),dp[lev].end(),L[u]))*ans[lev];
}
printf("%lld\n",res);
}
}
return 0;
}

ACM-ICPC 2018 沈阳赛区网络预赛 J. Ka Chang (树分块)的更多相关文章

  1. ACM-ICPC 2018 沈阳赛区网络预赛 J. Ka Chang (分块思想)

    题目链接:https://nanti.jisuanke.com/t/31451 题意: 给你一颗树,树上各点有初始权值,你有两种操作: 1. 给树中深度为l的点全部+x,(根节点为1,深度为0) 2. ...

  2. ACM-ICPC 2018 沈阳赛区网络预赛 J. Ka Chang(树状数组+分块)

    Given a rooted tree ( the root is node 1 ) of N nodes. Initially, each node has zero point. Then, yo ...

  3. ACM-ICPC 2018 沈阳赛区网络预赛 J Ka Chang

    Ka Chang 思路: dfs序+树状数组+分块 先dfs处理好每个节点的时间戳 对于每一层,如果这一层的节点数小于sqrt(n),那么直接按照时间戳在树状数组上更新 如果这一层节点个数大于sqrt ...

  4. ACM-ICPC 2018 沈阳赛区网络预赛 J. Ka Chang(树上分块+dfs序+线段树)

    题意 链接:https://nanti.jisuanke.com/t/A1998 给出一个有根树(根是1),有n个结点.初始的时候每个结点的值都是0.下面有q个操作,操作有两种,操作1.将深度为L(根 ...

  5. ACM-ICPC 2018 沈阳赛区网络预赛 J树分块

    J. Ka Chang Given a rooted tree ( the root is node 11 ) of NN nodes. Initially, each node has zero p ...

  6. ACM-ICPC 2018 徐州赛区网络预赛 J. Maze Designer (最大生成树+LCA求节点距离)

    ACM-ICPC 2018 徐州赛区网络预赛 J. Maze Designer J. Maze Designer After the long vacation, the maze designer ...

  7. ACM-ICPC 2018 沈阳赛区网络预赛 K Supreme Number(规律)

    https://nanti.jisuanke.com/t/31452 题意 给出一个n (2 ≤ N ≤ 10100 ),找到最接近且小于n的一个数,这个数需要满足每位上的数字构成的集合的每个非空子集 ...

  8. ACM-ICPC 2018 沈阳赛区网络预赛-K:Supreme Number

    Supreme Number A prime number (or a prime) is a natural number greater than 11 that cannot be formed ...

  9. ACM-ICPC 2018 沈阳赛区网络预赛-D:Made In Heaven(K短路+A*模板)

    Made In Heaven One day in the jail, F·F invites Jolyne Kujo (JOJO in brief) to play tennis with her. ...

随机推荐

  1. jQuery的end()方法使用详解

    end()方法的定义和用法: end()方法能够回到最近的一个"破坏性"操作之前,即将匹配的元素列表变为前一次的状态.如果没有破坏性操作将返回一个空集.破坏性操作的概念:指任何改变 ...

  2. 表空间_暂时表空间引起的错误:ora-01652 小例

    原创作品,出自 "深蓝的blog" 博客,深蓝的blog:http://blog.csdn.net/huangyanlong/article/details/46888243 报暂 ...

  3. Woody的Python学习笔记3

    Python运算符 Python逻辑运算符 and布尔与-假设x为false.x and y返回false,否则它返回y的计算值. or 布尔或-假设x是true,它返回true.否则它返回y的计算值 ...

  4. Hibernate_day02--Hibernate事务操作_api使用

    Hibernate事务操作 事务相关概念 1 什么是事务 事务是操作中最基本的单元,表示一组操作要么都成功,有一个失败那么所有都失败.最典型的场景:银行转账 2 事务特性 原子性 一致性 隔离性 持久 ...

  5. 编程之美 set 7 求数组中的最长递增子序列

    解法 1. 假设在目标数组 array[] 的前 i 个元素中, 最长递增子序列的长度为 LIS[i] 那么状态转移方程为 LIS[i] = max(1, LIS[k]+1) array[i+1] & ...

  6. hdu1244(dp)

    简单dp Max Sum Plus Plus Plus Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (J ...

  7. 【BZOJ4071】[Apio2015]巴邻旁之桥 Treap

    [BZOJ4071][Apio2015]巴邻旁之桥 Description 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域 A 和区域 B. 每一块区域沿着河岸都建了恰好 1000000001 ...

  8. Python全栈day28(描述符应用)

    描述符的使用 python是弱类型语言,及参数的赋值没有类型限制,下面通过描述符机制来实现类型限制功能 描述符应用1.py class Typed: def __get__(self, instanc ...

  9. java常用队列分析

    一.ArrayBlockingQueue 首先看一段源码: public class ArrayBlockingQueue<E> extends AbstractQueue<E> ...

  10. PAT 甲级 1024 Palindromic Number

    1024. Palindromic Number (25) 时间限制 400 ms 内存限制 65536 kB 代码长度限制 16000 B 判题程序 Standard 作者 CHEN, Yue A ...