热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

【bzoj3730:震波】【动态树分治】

题意给出一棵树,点有点权,每次询问距离一个点不超过kkk的点的点权和,或者修改一个点的点权,强制在线。n≤100000n\

题意

给出一棵树,点有点权,每次询问距离一个点不超过kkk的点的点权和,或者修改一个点的点权,强制在线。
n≤100000n\le100000n100000


分析

把点分树建出来,然后对每个分治中心用树状数组维护到该点距离为定值的点权和,以及到他点分树上父亲距离为定值的点权和。查询的时候每次沿着父亲往上跳,在计算当前点贡献时需要减去上一个点所在子树的贡献。
时间复杂度O(nlog⁡2n)O(n\log^2n)O(nlog2n)


代码

#include
#include
#include
#include
#include
#includeconst int N=100005;int n,m,cnt,last[N],dep[N],dfn[N],top,rmq[N*2][20],lg[N*2],bin[20],c1[20][N],c2[20][N],fa[N],rt,sum,mx[N],a[N],beg[N],size[N],dd[N],tot[20],end[N];
bool vis[N];
struct edge{int to,next;}e[N*2];void addedge(int u,int v)
{e[++cnt].to=v;e[cnt].next=last[u];last[u]=cnt;e[++cnt].to=u;e[cnt].next=last[v];last[v]=cnt;
}void dfs(int x,int fa)
{rmq[++top][0]=x;dep[x]=dep[fa]+1;dfn[x]=top;for (int i=last[x];i;i=e[i].next)if (e[i].to!=fa) dfs(e[i].to,x),rmq[++top][0]=x;
}void get_rmq()
{for (int i&#61;1;i<&#61;top;i&#43;&#43;) lg[i]&#61;log(i)/log(2);bin[0]&#61;1;for (int i&#61;1;i<&#61;lg[top];i&#43;&#43;) bin[i]&#61;bin[i-1]*2;for (int j&#61;1;j<&#61;lg[top];j&#43;&#43;)for (int i&#61;1;i&#43;bin[j]-1<&#61;top;i&#43;&#43;)rmq[i][j]&#61;dep[rmq[i][j-1]]<dep[rmq[i&#43;bin[j-1]][j-1]]?rmq[i][j-1]:rmq[i&#43;bin[j-1]][j-1];
}int get_dis(int x,int y)
{int l&#61;std::min(dfn[x],dfn[y]),r&#61;std::max(dfn[x],dfn[y]),w&#61;lg[r-l&#43;1];return dep[x]&#43;dep[y]-2*std::min(dep[rmq[l][w]],dep[rmq[r-bin[w]&#43;1][w]]);
}void get_root(int x,int fa)
{size[x]&#61;1;mx[x]&#61;0;for (int i&#61;last[x];i;i&#61;e[i].next){if (e[i].to&#61;&#61;fa||vis[e[i].to]) continue;get_root(e[i].to,x);size[x]&#43;&#61;size[e[i].to];mx[x]&#61;std::max(mx[x],size[e[i].to]);}mx[x]&#61;std::max(mx[x],sum-size[x]);if (!rt||mx[x]<mx[rt]) rt&#61;x;
}void ins1(int d,int x,int y)
{while (x<&#61;n) c1[d][x]&#43;&#61;y,x&#43;&#61;x&(-x);
}void ins2(int d,int x,int y)
{while (x<&#61;n) c2[d][x]&#43;&#61;y,x&#43;&#61;x&(-x);
}int query1(int d,int x)
{if (!d) return 0;int ans&#61;0;while (x) ans&#43;&#61;c1[d][x],x-&#61;x&(-x);return ans;
}int query2(int d,int x)
{if (!d) return 0;int ans&#61;0;while (x) ans&#43;&#61;c2[d][x],x-&#61;x&(-x);return ans;
}void pre(int x,int fa,int p1,int p2)
{ins1(dd[p1],beg[p1]&#43;get_dis(p1,x),a[x]);if (p2) ins2(dd[p1],beg[p1]&#43;get_dis(p2,x)-1,a[x]);tot[dd[p1]]&#43;&#43;;size[x]&#61;1;for (int i&#61;last[x];i;i&#61;e[i].next)if (e[i].to!&#61;fa&&!vis[e[i].to]) pre(e[i].to,x,p1,p2),size[x]&#43;&#61;size[e[i].to];
}void solve(int x)
{vis[x]&#61;1;dd[x]&#61;dd[fa[x]]&#43;1;beg[x]&#61;tot[dd[x]]&#43;1;pre(x,0,x,fa[x]);end[x]&#61;tot[dd[x]];for (int i&#61;last[x];i;i&#61;e[i].next)if (!vis[e[i].to]){rt&#61;0;sum&#61;size[e[i].to];get_root(e[i].to,x);fa[rt]&#61;x;solve(rt);}
}int query(int x,int y)
{int ans&#61;0,ls&#61;0,now&#61;x;while (now){int d&#61;y-get_dis(x,now);if (d>&#61;0) ans&#43;&#61;query1(dd[now],std::min(beg[now]&#43;d,end[now]))-query1(dd[now],beg[now]-1);if (d>&#61;0) ans-&#61;query2(dd[ls],std::min(beg[ls]&#43;d-1,end[ls]))-query2(dd[ls],beg[ls]-1);ls&#61;now;now&#61;fa[now];}return ans;
}void modify(int x,int y)
{int del&#61;y-a[x],now&#61;x;a[x]&#61;y;while (now){ins1(dd[now],beg[now]&#43;get_dis(now,x),del);if (fa[now]) ins2(dd[now],beg[now]&#43;get_dis(fa[now],x)-1,del);now&#61;fa[now];}
}int main()
{scanf("%d%d",&n,&m);for (int i&#61;1;i<&#61;n;i&#43;&#43;) scanf("%d",&a[i]);for (int i&#61;1;i<n;i&#43;&#43;){int x,y;scanf("%d%d",&x,&y);addedge(x,y);}dfs(1,0);get_rmq();rt&#61;0;sum&#61;n;get_root(1,0);solve(rt);int ans&#61;0;while (m--){int op,x,y;scanf("%d%d%d",&op,&x,&y);x^&#61;ans;y^&#61;ans;if (!op) printf("%d\n",ans&#61;query(x,y));else modify(x,y);}return 0;
}

推荐阅读
  • 本文探讨了如何在给定整数N的情况下,找到两个不同的整数a和b,使得它们的和最大,并且满足特定的数学条件。 ... [详细]
  • 题目描述:给定n个半开区间[a, b),要求使用两个互不重叠的记录器,求最多可以记录多少个区间。解决方案采用贪心算法,通过排序和遍历实现最优解。 ... [详细]
  • 题目Link题目学习link1题目学习link2题目学习link3%%%受益匪浅!-----&# ... [详细]
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • Explore a common issue encountered when implementing an OAuth 1.0a API, specifically the inability to encode null objects and how to resolve it. ... [详细]
  • 技术分享:从动态网站提取站点密钥的解决方案
    本文探讨了如何从动态网站中提取站点密钥,特别是针对验证码(reCAPTCHA)的处理方法。通过结合Selenium和requests库,提供了详细的代码示例和优化建议。 ... [详细]
  • 在前两篇文章中,我们探讨了 ControllerDescriptor 和 ActionDescriptor 这两个描述对象,分别对应控制器和操作方法。本文将基于 MVC3 源码进一步分析 ParameterDescriptor,即用于描述 Action 方法参数的对象,并详细介绍其工作原理。 ... [详细]
  • 本文详细介绍了Akka中的BackoffSupervisor机制,探讨其在处理持久化失败和Actor重启时的应用。通过具体示例,展示了如何配置和使用BackoffSupervisor以实现更细粒度的异常处理。 ... [详细]
  • 扫描线三巨头 hdu1928hdu 1255  hdu 1542 [POJ 1151]
    学习链接:http:blog.csdn.netlwt36articledetails48908031学习扫描线主要学习的是一种扫描的思想,后期可以求解很 ... [详细]
  • 本文详细探讨了VxWorks操作系统中双向链表和环形缓冲区的实现原理及使用方法,通过具体示例代码加深理解。 ... [详细]
  • Explore how Matterverse is redefining the metaverse experience, creating immersive and meaningful virtual environments that foster genuine connections and economic opportunities. ... [详细]
  • 本文介绍如何使用Objective-C结合dispatch库进行并发编程,以提高素数计数任务的效率。通过对比纯C代码与引入并发机制后的代码,展示dispatch库的强大功能。 ... [详细]
  • 1.如何在运行状态查看源代码?查看函数的源代码,我们通常会使用IDE来完成。比如在PyCharm中,你可以Ctrl+鼠标点击进入函数的源代码。那如果没有IDE呢?当我们想使用一个函 ... [详细]
author-avatar
JIE9118_755
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有