题意

给定一棵 \(n\) 个节点的树,每个节点有点权。完成 \(q\) 个操作——操作分两种:修改点 \(x\) 的点权、查询与 \(x\) 距离小于等于 \(d\) 的权值总和。

\(1 \leq n,q \leq 10^5\)

思路

从最简单的情况分析——只有一次查询。当然一遍 \(O(n)\) 的 \(\text{dfs}\) 可以直接写,不过要用点分治写的话,\(\text{solve}\) 函数直接容斥一下就可以了。

如果多个询问呢?其实在回答关于点 \(x\) 的询问时,其实只需要计算管辖 \(x\) 的所有重心的答案。我们只需要将点分治的过程记录下来,查询只查管辖 \(x\) 的重心,就可以在 \(\log n\) 的复杂度内回答一次询问了。

具体的实现每道题略有区别,但具体思路大致相同。别忘了我们是从一次查询作优化,那么我们对于一个点,记录它到重心的距离;对于每个重心开一个数组,表示管辖范围内距离为 \(d\) 的节点权值总和,然后前缀和一下就变成了距离小于等于 \(d\) 的权值总和,由于还有容斥的部分,故符号也要记录。若节点 \(x\) 询问为 \(d\) ,对于某一级重心 \(C\) ,距离为 \(dis\) ,对应前缀和数组 \(A\) ,对应符号为 \(s\ (s\in\{1,-1\})\) ,那么 \(x\) 与 \(C\) 的贡献就是 \(s\cdot A[d-dis]\) 。

而带上修改其实也没什么区别,只要把前缀和换成树状数组,然后每次修改,对于某一级重心 \(C\) ,在树状数组的 \(dis\) 位置做修改即可。

动态点分治就是把点分治的过程用适当容器去维护的算法。

代码

#include<bits/stdc++.h>
#define FOR(i,x,y) for(int i=(x),i##END=(y);i<=i##END;++i)
#define DOR(i,x,y) for(int i=(x),i##END=(y);i>=i##END;--i)
typedef long long LL;
using namespace std;
const int N=1e5+5;
template<const int maxn,const int maxm>struct Linked_list
{
int head[maxn],to[maxm],nxt[maxm],tot;
Linked_list(){clear();}
void clear(){memset(head,-1,sizeof(head));tot=0;}
void add(int u,int v){to[++tot]=v,nxt[tot]=head[u],head[u]=tot;}
#define EOR(i,G,u) for(int i=G.head[u];~i;i=G.nxt[i])
};
struct FenwickTree
{
#define lowbit(x) ((x)&-(x))
vector<int>c;int n;
void build(int _n){c.clear();FOR(i,0,n=_n+1)c.push_back(0);}
void update(int k,int val){for(k++;k<=n;k+=lowbit(k))c[k]+=val;}
int query(int k){int res=0;for(k=min(k+1,n);k>0;k^=lowbit(k))res+=c[k];return res;}
#undef lowbit
};
Linked_list<N,N<<1>G;
FenwickTree FT[N*2];int Fc;
int Fid[N][45],dis[N][45],lv[N];bool sgn[N][45];
int sz[N];bool mark[N];
int pw[N],n,q; void CFS(int u,int f,int tot,int &C,int &Mi)
{
sz[u]=1;int res=0;
EOR(i,G,u)
{
int v=G.to[i];
if(v==f||mark[v])continue;
CFS(v,u,tot,C,Mi);
sz[u]+=sz[v];
res=max(res,sz[v]);
}
res=max(res,tot-sz[u]);
if(res<Mi)C=u,Mi=res;
}
void dfs_init(int u,int f,int D,bool s)
{
Fid[u][++lv[u]]=Fc,dis[u][lv[u]]=D,sgn[u][lv[u]]=s;
EOR(i,G,u)
{
int v=G.to[i];
if(v==f||mark[v])continue;
dfs_init(v,u,D+1,s);
}
}
int dfs_dep(int u,int f,int d)
{
int res=d;
EOR(i,G,u)
{
int v=G.to[i];
if(v==f||mark[v])continue;
res=max(res,dfs_dep(v,u,d+1));
}
return res;
}
void dac(int u,int tot)
{
int Mi=1e9;
CFS(u,0,tot,u,Mi);
mark[u]=1;
FT[++Fc].build(dfs_dep(u,0,0));
dfs_init(u,0,0,1);
EOR(i,G,u)
{
int v=G.to[i];
if(mark[v])continue;
FT[++Fc].build(dfs_dep(v,u,1));
dfs_init(v,u,1,0);
dac(v,sz[u]>sz[v]?sz[v]:tot-sz[u]);
}
} void update(int u,int val)
{
FOR(i,1,lv[u])
{
int v=Fid[u][i],w=dis[u][i];
FT[v].update(w,val);
}
}
int query(int u,int d)
{
int res=0;
FOR(i,1,lv[u])
{
int v=Fid[u][i],w=dis[u][i];bool s=sgn[u][i];
if(s)res+=FT[v].query(d-w);
else res-=FT[v].query(d-w);
}
return res;
} int main()
{
while(~scanf("%d%d",&n,&q))
{
G.clear();
FOR(i,1,n)scanf("%d",&pw[i]);
FOR(i,1,n-1)
{
int u,v;
scanf("%d%d",&u,&v);
G.add(u,v),G.add(v,u);
}
Fc=0;
memset(lv,0,sizeof(lv));
memset(mark,0,sizeof(mark));
dac(1,n);
FOR(i,1,n)update(i,pw[i]);
while(q--)
{
char str[5];int x,y;
scanf("%s%d%d",str,&x,&y);
if(str[0]=='!')
{
update(x,y-pw[x]);
pw[x]=y;
}
else if(str[0]=='?')printf("%d\n",query(x,y));
}
}
return 0;
}

HDU 4918 Query on the subtree(动态点分治+树状数组)的更多相关文章

  1. 【BZOJ-3730】震波 动态点分治 + 树状数组

    3730: 震波 Time Limit: 15 Sec  Memory Limit: 256 MBSubmit: 626  Solved: 149[Submit][Status][Discuss] D ...

  2. bzoj 4372 烁烁的游戏——动态点分治+树状数组

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4372 和 bzoj 3070 震波 是一个套路.注意区间修改的话,树状数组不能表示 dis ...

  3. bzoj 3730 震波——动态点分治+树状数组

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3730 查询一个点可以转化为查询点分树上自己到根的路径上每个点对应范围答案.可用树状数组 f ...

  4. bzoj 4372 烁烁的游戏 —— 动态点分治+树状数组

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4372 本以为和 bzoj3730 一样,可以直接双倍经验了: 但要注意一下,树状数组不能查询 ...

  5. bzoj 3730 震波 —— 动态点分治+树状数组

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3730 建点分树,每个点记两个树状数组,存它作为重心管辖的范围内,所有点到它的距离情况和到它在 ...

  6. luogu 5311 [Ynoi2011]D1T3 动态点分治+树状数组

    我这份代码已经奇怪到一定程度了~ 洛谷上一直 $TLE$,但是本地造了几个数据都过了. 简单说一下题解: 先建出来点分树. 对于每一个询问,在点分树中尽可能向上跳祖先,看是否能够处理这个询问. 找到最 ...

  7. BZOJ_3295_[Cqoi2011]动态逆序对_CDQ分治+树状数组

    BZOJ_3295_[Cqoi2011]动态逆序对_CDQ分治+树状数组 Description 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一 ...

  8. [BZOJ 3295] [luogu 3157] [CQOI2011]动态逆序对(树状数组套权值线段树)

    [BZOJ 3295] [luogu 3157] [CQOI2011] 动态逆序对 (树状数组套权值线段树) 题面 给出一个长度为n的排列,每次操作删除一个数,求每次操作前排列逆序对的个数 分析 每次 ...

  9. 【BZOJ 1901】【Zju 2112】 Dynamic Rankings 动态K值 树状数组套主席树模板题

    达神题解传送门:http://blog.csdn.net/dad3zz/article/details/50638360 说一下我对这个模板的理解: 看到这个方法很容易不知所措,因为动态K值需要套树状 ...

随机推荐

  1. ztree使用实例

    一.当某节点被选择或被取消选择时获取所有被选择的节点: <link rel="stylesheet" href="./static/libs/ztree/css/m ...

  2. C# Http文件上传下载

    C# Http文件下载公共类(支持断点续传) http://www.cnblogs.com/hayden/archive/2012/04/26/2472815.html C# Http方式下载文件到本 ...

  3. Ford VCM II Ford VCM2 Diagnostic Tool with Ford IDS v108 Installed On Laptop Ready to Use

    HOW to VCM2 Ford VCM II with Ford IDS v108 Work Well? VCM2 Ford VCM2 Ford diagnostic tool hot sale i ...

  4. hive 用户行为分析(活跃。启动,留存,回访,新增)的一些经典sql

    很简单的sql 用户分析语句 :只要自定义简单的udf函数 获取统计时间createdatms字段的使用的日历类 add方法 和simpledateformat 将long类型的 定义多个重载方法 获 ...

  5. scrapy_novel_python

    # _*_ coding:UTF _8_ from bs4 import BeautifulSoup import requests,sys class downloader(object): def ...

  6. EditPlus配置GTK

    --GCC GTK Compile-- 命令:D:\GCC\MinGW_RP_Green\bin\gcc.exe 参数:$(FileName) -w -o $(FileNameNoExt).exe - ...

  7. ltp-fcntl36 偶尔出现fail unexpected data offset 20928 value 94

    每次出错的都是和posix相关先把结论说了: fn_ofd_w和fn_ofd_r的SAFE_FCNTL参数F_OFD_SETLKW fn_posix_w和fn_posix_r的SAFE_FCNTL参数 ...

  8. win10常见问题处理办法

    1.当笔记本连接wifi时,提示,无internet,安全,而手机能正常连接wifi时: cmd(需管理员权限)执行命令 netsh winsock reset 出现已重置,重启电脑 解决方法 2.当 ...

  9. 02: docker高级篇

    1.1 Docker Compose 1.Docker Compose 介绍 1. Compose是一个定义和管理多容器的工具,使用Python语言编写. 2. 使用Compose配置文件描述多个容器 ...

  10. squid代理服务器安装和配置

    服务器版本:centos6.5 squid版本:3.1 Squid介绍 Squid是一个缓存Internet 数据的软件,其接收用户的下载申请,并自动处理所下载的数据.当一个用户想要下载一个主页时,可 ...