BZOJ-1036 树的统计Count 链剖线段树(模板)=(树链剖分+线段树)
潇爷昨天刚刚讲完。。。感觉得还可以。。。对着模板打了个模板。。。还是不喜欢用指针。。。。
1036: [ZJOI2008]树的统计Count
Time Limit: 10 Sec Memory Limit: 162 MB
Submit: 10559 Solved: 4258
[Submit][Status][Discuss]
Description
一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w。我们将以下面的形式来要求你对这棵树完成一些操作: I. CHANGE u t : 把结点u的权值改为t II. QMAX u v: 询问从点u到点v的路径上的节点的最大权值 III. QSUM u v: 询问从点u到点v的路径上的节点的权值和 注意:从点u到点v的路径上的节点包括u和v本身
Input
输入的第一行为一个整数n,表示节点的个数。接下来n – 1行,每行2个整数a和b,表示节点a和节点b之间有一条边相连。接下来n行,每行一个整数,第i行的整数wi表示节点i的权值。接下来1行,为一个整数q,表示操作的总数。接下来q行,每行一个操作,以“CHANGE u t”或者“QMAX u v”或者“QSUM u v”的形式给出。 对于100%的数据,保证1<=n<=30000,0<=q<=200000;中途操作中保证每个节点的权值w在-30000到30000之间。
Output
对于每个“QMAX”或者“QSUM”的操作,每行输出一个整数表示要求输出的结果。
Sample Input
4
1 2
2 3
4 1
4 2 1 3
12
QMAX 3 4
QMAX 3 3
QMAX 3 2
QMAX 2 3
QSUM 3 4
QSUM 2 1
CHANGE 1 5
QMAX 3 4
CHANGE 3 6
QMAX 3 4
QMAX 2 4
QSUM 3 4
Sample Output
4
1
2
2
10
6
5
6
5
16
HINT
Source
树的分治
题目标解树的分治,写起来很难。。。而且我也不会。。。。
把树链剖,存储个点的信息。然后hash到线段树上。。。线段树维护区间最大和区间求和即可。。。
模板题,就照着模板来了。。。。我居然也开始压代码了。。(QaQ)
详情分析:http://blog.csdn.net/DaD3zZ/article/details/50628472
代码= =:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define inf 1000000000
struct data1{int l,r,sum,max;}t[120005];
struct data2{int next,go;}edge[60005];
int tree[30005],pre[30005],end[30005],son[30005],fa[30005],data[30005],num[30005],top[30005],deep[30005];
int n,x,y,cnt,tot,Q;
char opt[10];
int max(int a,int b){if (a>b) return a;return b;}
int min(int a,int b){if (a>b) return b;return a;}
void addedge(int u,int v){edge[++cnt].go=v;edge[cnt].next=end[u];end[u]=cnt;}
void dfs_1(int now,int f,int d)
{
deep[now]=d;fa[now]=f;num[now]=1;
for (int i=end[now]; i; i=edge[i].next)
{
int go=edge[i].go;if (go==f) continue;
dfs_1(go,now,d+1);num[now]+=num[go];
if (!son[now] || num[go]>num[son[now]]) son[now]=go;
}
}//链剖dfs_1
void dfs_2(int now,int number)
{
top[now]=number;tree[now]=++tot;
pre[tree[now]]=now;
if (!son[now]) return;
dfs_2(son[now],number);
for (int i=end[now]; i; i=edge[i].next)
{
int go=edge[i].go;
if (go!=son[now] && go!=fa[now]) dfs_2(go,go);
}
}//链剖dfs_2
void updata(int now)
{
t[now].sum=t[now<<1].sum+t[now<<1|1].sum;
t[now].max=max(t[now<<1].max,t[now<<1|1].max);
}//线段树。。
void build(int now,int l,int r)
{
t[now].l=l;t[now].r=r;
if (l==r) {t[now].sum=t[now].max=data[pre[l]];return;}
int mid=(l+r)>>1;
build(now<<1,l,mid);build(now<<1|1,mid+1,r);
updata(now);
}
void insert(int now,int loc,int add)
{
if (t[now].l==t[now].r)
{
t[now].sum+=add*(t[now].r-t[now].l+1);
t[now].max+=add;return;
}
int mid=(t[now].r+t[now].l)>>1;
if (loc<=mid) insert(now<<1,loc,add);else insert(now<<1|1,loc,add);
updata(now);
}
int query_sum(int now,int L,int R)
{
if (L<=t[now].l && R>=t[now].r) return t[now].sum;
int mid=(t[now].r+t[now].l)>>1;int ans=0;
if (L<=mid) ans+=query_sum(now<<1,L,R);
if (mid<R) ans+=query_sum(now<<1|1,L,R);
updata(now);
return ans;
}
int query_max(int now,int L,int R)
{
if (L<=t[now].l && R>=t[now].r) return t[now].max;
int mid=(t[now].l+t[now].r)>>1;int ans=-inf;
if (L<=mid) ans=max(ans,query_max(now<<1,L,R));
if (mid<R) ans=max(ans,query_max(now<<1|1,L,R));
updata(now);
return ans;
}
int find_max(int x,int y)
{
int f1=top[x],f2=top[y],tmp,ans=-inf;
while (f1!=f2)
{
if (deep[f1]<deep[f2]) {tmp=f1;f1=f2;f2=tmp;tmp=x;x=y;y=tmp;}
ans=max(ans,query_max(1,tree[f1],tree[x]));
x=fa[f1];f1=top[x];
}
ans=max(ans,query_max(1,min(tree[x],tree[y]),max(tree[x],tree[y])));
return ans;
}//查询原树上信息的区间最大
int find_sum(int x,int y)
{
int f1=top[x],f2=top[y],tmp,ans=0;
while (f1!=f2)
{
if (deep[f1]<deep[f2]) {tmp=f1;f1=f2;f2=tmp;tmp=x;x=y;y=tmp;}
ans+=query_sum(1,tree[f1],tree[x]);
x=fa[f1];f1=top[x];
}
ans+=query_sum(1,min(tree[x],tree[y]),max(tree[x],tree[y]));
return ans;
}//查询原树上信息的区间和
int main()
{
scanf("%d",&n);
for (int i=1; i<=n-1; i++)
{
scanf("%d%d",&x,&y);
addedge(x,y);addedge(y,x);
}
for (int i=1; i<=n; i++)
scanf("%d",&data[i]);
dfs_1(1,0,1);dfs_2(1,1);
build(1,1,n);
scanf("%d",&Q);
while (Q)
{
Q--;
scanf("%s%d%d",&opt,&x,&y);
if (opt[0]=='C') insert(1,tree[x],y-data[x]),data[x]=y;
else if (opt[1]=='M') printf("%d\n",find_max(x,y));
else if (printf("%d\n",find_sum(x,y)));
}
return 0;
}
BZOJ-1036 树的统计Count 链剖线段树(模板)=(树链剖分+线段树)的更多相关文章
- 【BZOJ1036】[ZJOI2008] 树的统计Count(一道可怕的模板题:树剖+线段树)
点此看题面 题解 这真的只是一道模板题:一个树链剖分套上一个线段树(令我窒息的组合). 既然是模板题,那就直接上代码吧. 代码 #include<bits/stdc++.h> #defin ...
- BZOJ 1036: [ZJOI2008]树的统计Count [树链剖分]【学习笔记】
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 14302 Solved: 5779[Submit ...
- Bzoj 1036: [ZJOI2008]树的统计Count 树链剖分,LCT
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 11102 Solved: 4490[Submit ...
- BZOJ.1036 [ZJOI2008]树的统计Count ( 点权树链剖分 线段树维护和与最值)
BZOJ.1036 [ZJOI2008]树的统计Count (树链剖分 线段树维护和与最值) 题意分析 (题目图片来自于 这里) 第一道树链剖分的题目,谈一下自己的理解. 树链剖分能解决的问题是,题目 ...
- bzoj 1036 [ZJOI2008]树的统计Count(树链剖分,线段树)
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 10677 Solved: 4313[Submit ...
- BZOJ 1036: [ZJOI2008]树的统计Count( 树链剖分 )
树链剖分... 不知道为什么跑这么慢 = = 调了一节课啊跪.. ------------------------------------------------------------------- ...
- bzoj 1036: [ZJOI2008]树的统计Count 树链剖分+线段树
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 16294 Solved: 6645[Submit ...
- BZOJ 1036: [ZJOI2008]树的统计Count (树链剖分模板题)
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 14982 Solved: 6081[Submit ...
- bzoj 1036: [ZJOI2008]树的统计Count (树链剖分+线段树 点权)
1036: [ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 21194 Solved: 8589[Submit ...
- BZOJ 1036 [ZJOI2008]树的统计Count (树链剖分)(线段树单点修改)
[ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 14968 Solved: 6079[Submit][Stat ...
随机推荐
- localStorage实现购物车数量单价和总价实时同步(二)
利用localStorage实时显示购物车小计和总价页面显示: 和昨天的原理相同,本地存储同时实时循环计算总价之和,注意循环时候的先清空再计算 Success is getting what you ...
- “display:block-inline形式的Span或Div中添加文字后,导致Span或Div排版掉落、错位”的原因及解决方法
最近在使用3个span(或div)制作带圆角边框的按钮时,按照常识,把span的display设置成inline-block,这样就可以设置span的width和height了,很爽的~ 可是当我在中 ...
- window下查看TCP端口连接情况
window下查看TCP端口连接情况:netstat -ano -p tcp|findstr 10001
- 我的WCF摸爬滚打之路(1)
等了好久终于等到今天!盼了好久终于把梦实现……哈哈,仅以此歌词来庆祝我为期3天的wcf学习之路圆满结束. 今天写这个文章的目的在于记录一下我自己在学习WCF的时候碰到的一些问题,俗话说,好记心不如烂笔 ...
- Win7 Qt4.8.5+QtCreator2.8.0+mingw配置过程
1:安装包 百度盘下载链接: Mingw: :安装步骤 1.首先安装qt creator,双击qt-creator-windows-opensource-2.8.0,注意安装目录不要有空格和特殊字符, ...
- Linux常用指令---netstat(网络端口)
netstat命令用于显示与IP.TCP.UDP和ICMP协议相关的统计数据,一般用于检验本机各端口的网络连接情况.netstat是在内核中访问网络及相关信息的程序,它能提供TCP连接,TCP和UDP ...
- 实验二实验报告 20135324&&20135330
北京电子科技学院(BESTI) 实 验 报 告 课程: 深入理解计算机系统 班级: 1353 姓名: 杨舒雯 张若嘉 学号: 20135324 20135330 成绩: 指导教师: 娄嘉鹏 实验日期: ...
- 利用JS跨域做一个简单的页面访问统计系统
其实在大部分互联网web产品中,我们通常会用百度统计或者谷歌统计分析系统,通过在程序中引入特定的JS脚本,然后便可以在这些统计系统中看到自己网站页面具体的访问情况.但是有些时候,由于一些特殊情况,我们 ...
- memset的使用
今天写程序的时候用了memset这个函数,我知道他是关于清空指针的,设置为0.但我用的时候,没有注意到他是以字节为单位进行操作的,改了半天其他程序内容.要注意的是,memset是对字字进行操作,所以以 ...
- ASP.NET MVC5 + EF6 入门教程 (6) View中的Razor使用
文章来源: Slark.NET-博客园 http://www.cnblogs.com/slark/p/mvc-5-ef-6-get-started-model.html 上一节:ASP.NET MVC ...