HDU 4918 Query on the subtree(动态点分治+树状数组)
题意
给定一棵 \(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(动态点分治+树状数组)的更多相关文章
- 【BZOJ-3730】震波 动态点分治 + 树状数组
3730: 震波 Time Limit: 15 Sec Memory Limit: 256 MBSubmit: 626 Solved: 149[Submit][Status][Discuss] D ...
- bzoj 4372 烁烁的游戏——动态点分治+树状数组
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4372 和 bzoj 3070 震波 是一个套路.注意区间修改的话,树状数组不能表示 dis ...
- bzoj 3730 震波——动态点分治+树状数组
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3730 查询一个点可以转化为查询点分树上自己到根的路径上每个点对应范围答案.可用树状数组 f ...
- bzoj 4372 烁烁的游戏 —— 动态点分治+树状数组
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4372 本以为和 bzoj3730 一样,可以直接双倍经验了: 但要注意一下,树状数组不能查询 ...
- bzoj 3730 震波 —— 动态点分治+树状数组
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3730 建点分树,每个点记两个树状数组,存它作为重心管辖的范围内,所有点到它的距离情况和到它在 ...
- luogu 5311 [Ynoi2011]D1T3 动态点分治+树状数组
我这份代码已经奇怪到一定程度了~ 洛谷上一直 $TLE$,但是本地造了几个数据都过了. 简单说一下题解: 先建出来点分树. 对于每一个询问,在点分树中尽可能向上跳祖先,看是否能够处理这个询问. 找到最 ...
- BZOJ_3295_[Cqoi2011]动态逆序对_CDQ分治+树状数组
BZOJ_3295_[Cqoi2011]动态逆序对_CDQ分治+树状数组 Description 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一 ...
- [BZOJ 3295] [luogu 3157] [CQOI2011]动态逆序对(树状数组套权值线段树)
[BZOJ 3295] [luogu 3157] [CQOI2011] 动态逆序对 (树状数组套权值线段树) 题面 给出一个长度为n的排列,每次操作删除一个数,求每次操作前排列逆序对的个数 分析 每次 ...
- 【BZOJ 1901】【Zju 2112】 Dynamic Rankings 动态K值 树状数组套主席树模板题
达神题解传送门:http://blog.csdn.net/dad3zz/article/details/50638360 说一下我对这个模板的理解: 看到这个方法很容易不知所措,因为动态K值需要套树状 ...
随机推荐
- 用 hashcat 破解 WIFI WPA2破解
首先用CDlinux系统进行抓包,CDlinux抓包我就不详细说明 到这里可以查看如何安装CDlinux http://jingyan.baidu.com/article/7f766daf5173a9 ...
- 20165316 预备作业3 Linux安装及学习
Linux安装 我下载的是VirtualBox 5.2.6和Ubuntu 17.10.1,感觉这两个版本的兼容性不是太好,因为我在Oracle的官网社区中看到不少新版本的问题没有得到解决,而老版本(V ...
- 谷歌重磅开源强化学习框架Dopamine吊打OpenAI
谷歌重磅开源强化学习框架Dopamine吊打OpenAI 近日OpenAI在Dota 2上的表现,让强化学习又火了一把,但是 OpenAI 的强化学习训练环境 OpenAI Gym 却屡遭抱怨,比如不 ...
- JustOj 2039: 成绩排名 (结构体排序)
题目描述 每次期末考试成绩出来之前的一段时间大豪哥心里都是痛苦的,总感觉自己会在班上排名特别差.所以当成绩出来以后大豪哥想快点知道班上的总排名,以便知道自己的排名.(PS:大豪哥班上有个学霸名叫日天, ...
- Codeforce 296A - Yaroslav and Permutations
Yaroslav has an array that consists of n integers. In one second Yaroslav can swap two neighboring a ...
- ID3和C4.5分类决策树算法 - 数据挖掘算法(7)
(2017-05-18 银河统计) 决策树(Decision Tree)是在已知各种情况发生概率的基础上,通过构成决策树来判断其可行性的决策分析方法,是直观运用概率分析的一种图解法.由于这种决策分支画 ...
- The Little Prince-11/28
The Little Prince-11/28 Today I find some beautiful words from the book. You know -- one loves the s ...
- win10系统jdk安装和环境变量配置
新换电脑的原因,要重新安装jdk,完整记录一下安装过程 jdk版本用的1.7(公司默认版本) 这是jdk安装目录 更改为D:\jdk\java\jdk1.7 安装jre目录 更改为D:\jdk\ ...
- 深度估计&平面检测小结
https://yq.aliyun.com/ziliao/582885 最近一段时间已知忙着赶图像分析与理解的项目,在三个星期内强行接触了CNN,MRF,Caffe,openCV在内的很多东西.现在项 ...
- STM32开发 -- 4G模块开发详解(转)
STM32开发 -- 4G模块开发详解(1) STM32开发 -- 4G模块开发详解(2) STM32开发 -- 4G模块开发详解(3) STM32开发 -- 4G模块开发详解(4)