[bzoj] 1036 Count
原题
树链剖分板子题
树剖详解:
#include<cstdio>
#include<algorithm>
typedef long long ll;
#define N 30010
using namespace std;
int n,x,y,m,a[N],f[N],dfn[N],deep[N],head[N],cnt=1,tp[N],ref[N],t,son[N],size[N];
char s[10];
struct hhh
{
int to,next;
}edge[2*N];
struct node
{
int l,r,data,mx;
}tre[4*N];
int read()
{
int ans=0,fu=1;
char j=getchar();
for (;(j<'0' || j>'9') && j!='-';j=getchar()) ;
if (j=='-') fu=-1,j=getchar();
for (;j>='0' && j<='9';j=getchar()) ans*=10,ans+=j-'0';
return ans*fu;
}
void add(int u,int v)
{
edge[cnt].to=v;
edge[cnt].next=head[u];
head[u]=cnt++;
}
void dfs1(int x,int fa,int dep)
{
f[x]=fa;
deep[x]=dep+1;
int mx=0;
for (int i=head[x],v;i;i=edge[i].next)
{
v=edge[i].to;
if (v!=fa)
{
dfs1(v,x,dep+1);
size[x]+=size[v];
if (size[v]>mx) son[x]=v,mx=size[v];
}
}
size[x]++;
}
void dfs2(int x,int top)
{
dfn[x]=++t;
ref[t]=x;
tp[x]=top;
if (son[x]) dfs2(son[x],top);
for (int i=head[x],v;i;i=edge[i].next)
{
v=edge[i].to;
if (v!=son[x] && v!=f[x])
dfs2(v,v);
}
}
void build(int i,int l,int r)
{
tre[i].l=l;
tre[i].r=r;
if (l==r)
{
tre[i].data=a[ref[l]];
tre[i].mx=a[ref[l]];
return ;
}
int mid=(l+r)>>1;
build(i*2,l,mid);
build(i*2+1,mid+1,r);
tre[i].data=tre[i*2].data+tre[i*2+1].data;
tre[i].mx=max(tre[i*2].mx,tre[i*2+1].mx);
}
void modify(int i,int x,int y)
{
if (tre[i].l==x && tre[i].l==tre[i].r)
{
tre[i].data=y;
tre[i].mx=y;
return ;
}
int mid=(tre[i].l+tre[i].r)>>1;
if (x>mid) modify(i*2+1,x,y);
else modify(i*2,x,y);
tre[i].data=tre[i*2].data+tre[i*2+1].data;
tre[i].mx=max(tre[i*2].mx,tre[i*2+1].mx);
}
ll query(int i,int l,int r,int p)
{
if (tre[i].l==l && tre[i].r==r)
if (p) return tre[i].data;
else return tre[i].mx;
int mid=(tre[i].l+tre[i].r)>>1;
if (l>mid) return query(i*2+1,l,r,p);
else if (r<=mid) return query(i*2,l,r,p);
else if (p) return query(i*2,l,mid,p)+query(i*2+1,mid+1,r,p);
else return max(query(i*2,l,mid,p),query(i*2+1,mid+1,r,p));
}
ll pathquery(int u,int v,int p)
{
ll ans=0,tmp=-1000000000;
while (tp[u]!=tp[v])
{
if (deep[tp[u]]<deep[tp[v]]) swap(u,v);
if (p) ans+=query(1,dfn[tp[u]],dfn[u],p);
else tmp=max(tmp,query(1,dfn[tp[u]],dfn[u],p));
u=f[tp[u]];
}
if (deep[u]>deep[v]) swap(u,v);
if (p) return ans+query(1,dfn[u],dfn[v],p);
else return max(tmp,query(1,dfn[u],dfn[v],p));
}
int main()
{
n=read();
for (int i=1;i<n;i++)
{
x=read();
y=read();
add(x,y);
add(y,x);
}
for (int i=1;i<=n;i++) a[i]=read();
dfs1(1,0,0);
dfs2(1,1);
build(1,1,n);
m=read();
while (m--)
{
scanf("%s",s);
x=read();
y=read();
if (s[1]=='M')
printf("%lld\n",pathquery(x,y,0));
else if (s[1]=='S')
printf("%lld\n",pathquery(x,y,1));
else modify(1,dfn[x],y);
}
return 0;
}
[bzoj] 1036 Count的更多相关文章
- BZOJ.1036 [ZJOI2008]树的统计Count ( 点权树链剖分 线段树维护和与最值)
BZOJ.1036 [ZJOI2008]树的统计Count (树链剖分 线段树维护和与最值) 题意分析 (题目图片来自于 这里) 第一道树链剖分的题目,谈一下自己的理解. 树链剖分能解决的问题是,题目 ...
- [BZOJ 1036] [ZJOI2008] 树的统计Count 【Link Cut Tree】
题目链接:BZOJ - 1036 题目分析 这道题可以用树链剖分,块状树等多种方法解决,也可以使用 LCT. 修改某个点的值时,先将它 Splay 到它所在的 Splay 的根,然后修改它的值,再将它 ...
- BZOJ 1036: [ZJOI2008]树的统计Count [树链剖分]【学习笔记】
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 14302 Solved: 5779[Submit ...
- BZOJ 1036: [ZJOI2008]树的统计Count
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MB Submit: 14354 Solved: 5802 [Subm ...
- bzoj 1036 [ZJOI2008]树的统计Count(树链剖分,线段树)
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 10677 Solved: 4313[Submit ...
- Bzoj 1036: [ZJOI2008]树的统计Count 树链剖分,LCT
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 11102 Solved: 4490[Submit ...
- 数据结构(LCT动态树):BZOJ 1036: [ZJOI2008]树的统计Count
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 12266 Solved: 4945[Submit ...
- BZOJ 1036: [ZJOI2008]树的统计Count( 树链剖分 )
树链剖分... 不知道为什么跑这么慢 = = 调了一节课啊跪.. ------------------------------------------------------------------- ...
- bzoj 1036: [ZJOI2008]树的统计Count 树链剖分+线段树
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 16294 Solved: 6645[Submit ...
随机推荐
- js一键复制到剪切板
<div id='demo'>我就是被复制的内容<span>点击复制<span></div> $('#demo').on('click','span', ...
- [转]C++ explicit的作用
explicit作用: 在C++中,explicit关键字用来修饰类的构造函数,被修饰的构造函数的类,不能发生相应的隐式类型转换,只能以显示的方式进行类型转换. explicit使用注意事项: * e ...
- poj_1730_Perfect Pth Powers
We say that x is a perfect square if, for some integer b, x = b 2. Similarly, x is a perfect cube if ...
- Python学习之set集合
set集合以{}保存一组可迭代对象,如列表,字符串,set集合本身.集合内的元素若有重复的,将自动去除重复元素 a=set([1,2,3]) print(a) b=set('hello python' ...
- 30-RoutingMiddleware介绍以及MVC引入
1-构建路由 public class Startup { // This method gets called by the runtime. Use this method to add serv ...
- talent-aio源码阅读小记(二)
我们上一篇提到了talent-aio的四类Task:DecodeRunnable.HandlerRunnable.SendRunnable.CloseRunnable,并且分析了这些task的基类Ab ...
- js双轴柱状图
<!doctype html><html lang="en"><head> <script type="text/javascr ...
- MVC WebAPI 的基本使用
1.什么是WebAPI Web API是网络应用程序接口.包含了广泛的功能,网络应用通过API接口,可以实现存储服务.消息服务.计算服务等能力,利用这些能力可以进行开发出强大功能的web应用. 它可以 ...
- 110Balanced Binary Tree
问题:判断二叉树是否为平衡二叉树分析:树上的任意结点的左右子树高度差不超过1,则为平衡二叉树. 搜索递归,记录i结点的左子树高度h1和右子树高度h2,则i结点的高度为max(h1,h2 ...
- hive原理
什么是Hive Hive是基于Hadoop的一个数据仓库工具,可以将结构化的数据文件映射为一张数据库表,并提供类SQL查询功能. Hive架构图 Jobtracker是hadoop1.x中的组件,它的 ...