题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3730

建点分树,每个点记两个树状数组,存它作为重心管辖的范围内,所有点到它的距离情况和到它在点分树上的父亲的距离情况;

于是算的时候可以减去重复的,就是跳到父亲之前把自己会被重复统计的部分减去;

注意跳点分树父亲时,查询的距离都是原本询问点到那个父亲的距离,而不是上一层父亲到那个父亲的距离;

树状数组的大小总共是 nlogn 的,因为每层有 n 个点,一共 logn 层;

于是一开始写的是开一个长长的树状数组,记录每个点用的是它上面的哪一段;

然后感觉很艰难...段的长度到底应该是最大深度还是点数?第二个树状数组的段长度是2倍?......

于是改成 vector 了,用 resize() 函数可以方便地开大小,虽然还是不知道具体应该开多大?

查询距离可以把求 LCA 的过程用欧拉序+ST表变成 O(1) 的;

艰难写好,对拍竟然没问题!然而交上去却一直RE...

后来发现前一个操作是修改,并不需要把 ans 赋成0!(对拍没发现是因为不会生成数据,只好每次只做一次查询)

然而还是RE...??

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
using namespace std;
int const xn=1e5+,xm=xn*,xxn=(xn<<)+;
int n,hd[xn],ct,to[xn<<],nxt[xn<<],v[xn];
int dep[xn],fa[xn],siz[xn],mx,rt,tmp,ans;
//int t[xm],t2[xm<<1],wmx,wmx2,l[xn],r[xn],l2[xn],r2[xn];
int tim,in[xn],st[xxn][],bit[xxn],bin[];
bool vis[xn];
vector<int>tr[xn],tr2[xn];
int rd()
{
int ret=,f=; char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=; ch=getchar();}
while(ch>=''&&ch<='')ret=ret*+ch-'',ch=getchar();
return f?ret:-ret;
}
int Max(int x,int y){return x>y?x:y;}
int Min(int x,int y){return x<y?x:y;}
void add(int x,int y){to[++ct]=y; nxt[ct]=hd[x]; hd[x]=ct;}
void dfsx(int x,int ff)
{
dep[x]=dep[ff]+; in[x]=++tim; st[tim][]=x;
for(int i=hd[x],u;i;i=nxt[i])
if((u=to[i])!=ff)dfsx(u,x),st[++tim][]=x;
//out[x]=++tim; st[tim][0]=x;//
}
void init()
{
dfsx(,);
bin[]=; for(int i=;i<=;i++)bin[i]=bin[i-]*;
bit[]=; for(int i=;i<=tim;i++)bit[i]=bit[i>>]+;
for(int i=;i<=;i++)
for(int j=;j+bin[i-]<=tim&&st[j+bin[i-]][i-];j++)
st[j][i]=Min(st[j][i-],st[j+bin[i-]][i-]);
}
int lca(int x,int y)
{
if(in[x]>in[y])swap(x,y);
int t=bit[in[y]-in[x]+];
return Min(st[in[x]][t],st[in[y]-bin[t]+][t]);
}
int dist(int x,int y){return dep[x]+dep[y]-*dep[lca(x,y)];}
void getrt(int x,int ff,int sum,int dis)
{
int nmx=; siz[x]=;
for(int i=hd[x],u;i;i=nxt[i])
{
if((u=to[i])==ff||vis[u])continue;
getrt(u,x,sum,dis+); siz[x]+=siz[u];
nmx=Max(nmx,siz[u]);
}
nmx=Max(nmx,sum-siz[x]);
if(nmx<mx)mx=nmx,rt=x,tmp=dis;
}
/*
void ins(int nw,int x,int v){if(x==0){t[l[nw]]+=v; return;} for(int p=l[nw]+x;p<=r[nw];x+=(x&-x),p=l[nw]+x)t[p]+=v;}
int query(int nw,int x){int ret=0; for(int p=l[nw]+x;x;x-=(x&-x),p=l[nw]+x)ret+=t[p]; return ret+t[l[nw]];}
void ins2(int nw,int x,int v){
if(x==0){t2[l2[nw]]+=v; return;}
printf("nw=%d x=%d v=%d\n",nw,x,v);
for(int p=l2[nw]+x;p<=r2[nw];x+=(x&-x),p=l2[nw]+x)t2[p]+=v,printf("t2[%d]+=%d x=%d\n",p,v,x);}
int query2(int nw,int x){int ret=0; for(int p=l2[nw]+x;x;x-=(x&-x),p=l2[nw]+x)ret+=t2[p],printf("t2[%d]=%d x=%d\n",p,t2[p],x); return ret+t2[l[nw]];}
*/
void ins(int nw,int x,int v){if(x==){tr[nw][]+=v; return;} for(;x<tr[nw].size();x+=(x&-x))tr[nw][x]+=v;}
int query(int nw,int x){int ret=; x=Min(x,tr[nw].size()-); for(;x;x-=(x&-x))ret+=tr[nw][x]; return ret+tr[nw][];}
void ins2(int nw,int x,int v){if(x==){tr2[nw][]+=v; return;} for(;x<tr2[nw].size();x+=(x&-x))tr2[nw][x]+=v;}
int query2(int nw,int x){int ret=; x=Min(x,tr2[nw].size()-); for(;x;x-=(x&-x))ret+=tr2[nw][x]; return ret+tr2[nw][];}
void dfs(int x,int ff,int dis)
{
ins(rt,dis,v[x]); if(fa[rt])ins2(rt,dist(x,fa[rt]),v[x]);
for(int i=hd[x],u;i;i=nxt[i])
if((u=to[i])!=ff&&!vis[u])dfs(u,x,dis+);
}
void work(int x,int sum)
{
vis[x]=;
ins(x,,v[x]); if(fa[x])ins2(x,dist(x,fa[x]),v[x]);//
for(int i=hd[x],u;i;i=nxt[i])
{
if(vis[u=to[i]])continue;
dfs(u,x,);
}
for(int i=hd[x],u;i;i=nxt[i])
{
if(vis[u=to[i]])continue;
int ns=(siz[u]>siz[x]?sum-siz[x]:siz[u]);
mx=xn; getrt(u,,ns,);
fa[rt]=x;
//len[rt]=tmp;
/*
l[rt]=wmx+1; wmx+=mx+1; r[rt]=wmx;//maxdep
l2[rt]=wmx2+1; wmx2+=2*mx+2; r2[rt]=wmx2;
*/
tr[rt].resize(mx+); tr2[rt].resize(*mx+);
work(rt,ns);
}
}
void ask(int p,int x,int k,int dis)
{
ans+=query(x,k); int len=dist(p,fa[x]);
if(dis>=len&&fa[x])
{
ans-=query2(x,dis-len);
ask(p,fa[x],dis-len,dis);
}
}
void change(int p,int x,int v1,int v2)
{
ins(x,dist(p,x),-v1); ins2(x,dist(p,fa[x]),-v1);
ins(x,dist(p,x),v2); ins2(x,dist(p,fa[x]),v2);
if(fa[x])change(p,fa[x],v1,v2);
}
int main()
{
n=rd(); int m=rd();
for(int i=;i<=n;i++)v[i]=rd();
for(int i=,x,y;i<n;i++)x=rd(),y=rd(),add(x,y),add(y,x);
init();
mx=xn; getrt(,,n,);
/*
l[rt]=wmx+1; wmx+=mx+1; r[rt]=wmx;
l2[rt]=wmx2+1; wmx2+=2*mx+2; r2[rt]=wmx2;
*/
tr[rt].resize(mx+); tr2[rt].resize(*mx+);
work(rt,n);
for(int i=,op,x,y;i<=m;i++)
{
op=rd(); x=(rd()^ans); y=(rd()^ans);
if(op==)ans=,ask(x,x,y,y),printf("%d\n",ans);
else change(x,x,v[x],y),v[x]=y;//,ans=0;
}
return ;
}

RE

然后借鉴了TJ:https://www.cnblogs.com/enigma-aw/p/6209545.html

预处理父亲好方便!

代码如下:

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
using namespace std;
int const xn=1e5+;
int n,hd[xn],ct,to[xn<<],nxt[xn<<],v[xn];
int dep[xn],siz[xn],mx,rt,ans;
int fa[xn][],dis[xn][];
bool vis[xn];
vector<int>tr[xn],tr2[xn];
int rd()
{
int ret=,f=; char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=; ch=getchar();}
while(ch>=''&&ch<='')ret=ret*+ch-'',ch=getchar();
return f?ret:-ret;
}
int Max(int x,int y){return x>y?x:y;}
int Min(int x,int y){return x<y?x:y;}
void add(int x,int y){to[++ct]=y; nxt[ct]=hd[x]; hd[x]=ct;}
void getrt(int x,int ff,int sum)
{
int nmx=; siz[x]=;
for(int i=hd[x],u;i;i=nxt[i])
{
if((u=to[i])==ff||vis[u])continue;
getrt(u,x,sum); siz[x]+=siz[u];
nmx=Max(nmx,siz[u]);
}
nmx=Max(nmx,sum-siz[x]);
if(nmx<mx)mx=nmx,rt=x;
}
void build(int x,int p,int ff,int d)
{
for(int i=hd[x],u;i;i=nxt[i])
{
if((u=to[i])==ff||vis[u])continue;
fa[u][++dep[u]]=p; dis[u][dep[u]]=d;
build(u,p,x,d+);
}
}
void work(int x,int sum)
{
vis[x]=; build(x,x,,);
tr[x].resize(sum+); tr2[x].resize(sum+);//nlogn
for(int i=hd[x],u;i;i=nxt[i])
{
if(vis[u=to[i]])continue;
int ns=(siz[u]>siz[x]?sum-siz[x]:siz[u]);
mx=xn; getrt(u,,ns); work(rt,ns);
}
}
void ins(int nw,int x,int v){for(;x<tr[nw].size()&&x;x+=(x&-x))tr[nw][x]+=v;}
int query(int nw,int x){if(x<)return ; int ret=; x=Min(x,tr[nw].size()-); for(;x;x-=(x&-x))ret+=tr[nw][x]; return ret+v[nw];}//v[nw]
void ins2(int nw,int x,int v){for(;x<tr2[nw].size()&&x;x+=(x&-x))tr2[nw][x]+=v;}//x
int query2(int nw,int x){if(x<)return ; int ret=; x=Min(x,tr2[nw].size()-); for(;x;x-=(x&-x))ret+=tr2[nw][x]; return ret;}
int ask(int x,int k)
{
int ret=query(x,k);
for(int i=dep[x];i;i--)//
if(k>=dis[x][i])
ret+=query(fa[x][i],k-dis[x][i])-query2(fa[x][i+],k-dis[x][i]);
return ret;
}
void change(int x,int val)
{
int d=dis[x][dep[x]],ff; ins2(x,d,val);//x
for(int i=dep[x];i;i--)
{
d=dis[x][i]; ff=fa[x][i]; ins(ff,d,val);
d=dis[x][i-]; ins2(ff,d,val);//fa[x][i]
}
}
int main()
{
n=rd(); int m=rd();
for(int i=;i<=n;i++)v[i]=rd();
for(int i=,x,y;i<n;i++)x=rd(),y=rd(),add(x,y),add(y,x);
mx=xn; getrt(,,n); work(rt,n);
for(int i=;i<=n;i++)fa[i][dep[i]+]=i;
for(int i=;i<=n;i++)change(i,v[i]);
for(int i=,op,x,y;i<=m;i++)
{
op=rd(); x=(rd()^ans); y=(rd()^ans);
if(op==)ans=ask(x,y),printf("%d\n",ans);
else change(x,y-v[x]),v[x]=y;//,ans=0;
}
return ;
}

bzoj 3730 震波 —— 动态点分治+树状数组的更多相关文章

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

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

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

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

  3. BZOJ 2683 简单题 cdq分治+树状数组

    题意:链接 **方法:**cdq分治+树状数组 解析: 首先对于这道题,看了范围之后.二维的数据结构是显然不能过的.于是我们可能会考虑把一维排序之后还有一位上数据结构什么的,然而cdq分治却可以非常好 ...

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

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

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

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

  6. HDU 4918 Query on the subtree(动态点分治+树状数组)

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

  7. BZOJ 3648: 寝室管理( 点分治 + 树状数组 )

    1棵树的话, 点分治+你喜欢的数据结构(树状数组/线段树/平衡树)就可以秒掉, O(N log^2 N). 假如是环套树, 先去掉环上1条边, 然后O(N log^2 N)处理树(同上); 然后再O( ...

  8. BZOJ 1176: [Balkan2007]Mokia( CDQ分治 + 树状数组 )

    考虑cdq分治, 对于[l, r)递归[l, m), [m, r); 然后计算[l, m)的操作对[m, r)中询问的影响就可以了. 具体就是差分答案+排序+离散化然后树状数组维护.操作数为M的话时间 ...

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

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

随机推荐

  1. J - Max Sum

    J - Max Sum Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u Descrip ...

  2. tomcat日志按天切分

    1. 下载工具cronolog wget http://cronolog.org/download/cronolog-1.6.2.tar.gz 这是网上流传的下载地址,好像没用,所以需要自己去网上找. ...

  3. Future Promise 模式(netty源码9)

    netty源码死磕9  Future Promise 模式详解 1. Future/Promise 模式 1.1. ChannelFuture的由来 由于Netty中的Handler 处理都是异步IO ...

  4. The space of such functions is known as a reproducing kernel Hilbert space.

    Reproducing kernel Hilbert space Mapping the points to a higher dimensional feature space http://www ...

  5. 6.让ORM映射执行的时候打印SQL语句

    配置Django日志:\hello_django\hello_django\settings.py 文件中的 LOGGING 加入如下配置: LOGGING = { 'version': 1, 'di ...

  6. PHP 提高PHP性能的编码技巧以及性能优化

    0.用单引号代替双引号来包含字符串,这样做会更快一些.因为PHP会在双引号包围的字符串中搜寻变量,单引号则不会,注意:只有echo能这 么做,它是 一种可以把多个字符串当作参数的“函数”(译注:PHP ...

  7. You are using pip version 8.1.2, however version 9.0.1 is available.

    [root@localhost ~]# pip install virtualenvmapperCollecting virtualenvmapper  Could not find a versio ...

  8. date_default_timezone_get():

    [Symfony\Component\Debug\Exception\ContextErrorException]                      Warning: date_default ...

  9. awk 字符串函数

    awk 提供了许多强大的字符串函数,见下表: awk 内置字符串函数 gsub(r,s) 在整个 $0 中用 s 替代 r gsub(r,s,t) 在整个 t 中用 s 替代 r index(s,t) ...

  10. HDU - 5703 Desert 【找规律】

    题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=5703 题意 给出一杯容量为N的水 每次至少喝1个单位 有多少种不同的方式喝完 比如 给出3 就有4种方 ...