HYSBZ 1036 树的统计Count(树链剖分)题解
思路:
树链剖分,不知道说什么...我连模板都不会用
代码:
#include<map>
#include<ctime>
#include<cmath>
#include<stack>
#include<queue>
#include<string>
#include<vector>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#define ll long long
const int maxn = 60000+5;
const int INF = 0x3f3f3f3f;
using namespace std;
int n,m,tol;
int cnt;
struct Node{
int u,v,next;
}node[maxn << 1];
int head[maxn];
int fa[maxn],siz[maxn],dep[maxn],son[maxn],id[maxn],top[maxn];
//父节点,子树大小,深度,重儿子,dfs执行顺序,重链的起始点
void addnode(int u,int v){
node[tol].u = u;
node[tol].v = v;
node[tol].next = head[u];
head[u] = tol++;
}
void dfs1(int u,int father,int depth){ //确定fa,siz,dep,son
son[u] = 0;
dep[u] = depth;
fa[u] = father;
siz[u] = 1;
for(int i = head[u];~i;i = node[i].next){
int v = node[i].v;
if(v == father) continue; //储存无向边,需判断
dfs1(v,u,depth + 1);
siz[u] += siz[v];
if(siz[v] > siz[son[u]]) //更新重儿子
son[u] = v;
}
}
void dfs2(int u,int tp){ //按dfs执行顺序编号
top[u] = tp; //重链的起始点
id[u] = ++cnt; //编号
if(!son[u]) return;
//有重儿子,是重链,继续往下
dfs2(son[u],tp);
for(int i = head[u];~i;i = node[i].next){
int v = node[i].v;
if(v == fa[u] || v == son[u]) continue;
dfs2(v,v);
}
}
/*
线段树
*/
int sum[maxn<<2],mx[maxn<<2];
void push_up(int rt){
sum[rt] = sum[rt << 1] + sum[rt << 1 | 1];
mx[rt] = max(mx[rt << 1],mx[rt << 1 | 1]);
}
/*void build(int l,int r,int rt){
sum[rt] = 0;
mx[rt] = -INF; //注意
if(l == r){
sum[rt] = mx[rt] = val[l];
return;
}
int m = (l + r) >> 1;
build(l,m,rt << 1);
build(m + 1,r,rt << 1 | 1);
push_up(rt);
}*/
void update(int pos,int val,int l,int r,int rt){
if(l == r){
sum[rt] = mx[rt] = val;
return;
}
int m = (l + r) >> 1;
if(pos <= m)
update(pos,val,l,m,rt << 1);
else
update(pos,val,m + 1,r,rt << 1 | 1);
push_up(rt);
}
int queryS(int L,int R,int l,int r,int rt){
if(L <= l && R >= r){
return sum[rt];
}
int m = (l + r) >> 1;
int ans = 0;
if(L <= m)
ans += queryS(L,R,l,m,rt << 1);
if(R > m)
ans += queryS(L,R,m + 1,r,rt << 1 | 1);
return ans;
}
int queryM(int L,int R,int l,int r,int rt){
if(L <= l && R >= r){
return mx[rt];
}
int m = (l + r) >> 1;
int ans = -INF;
if(L <= m)
ans = max(ans,queryM(L,R,l,m,rt << 1));
if(R > m)
ans = max(ans,queryM(L,R,m + 1,r,rt << 1 | 1));
return ans;
}
//////////////////////////////////////////
int FindS(int u,int v){
int ans = 0;
while(top[u] != top[v]){ //不在同一条链上,就从u所在转移到v所在
if(dep[top[u]] < dep[top[v]]){
swap(u,v);
}
ans += queryS(id[top[u]],id[u],1,n,1);
u = fa[top[u]];
}
if(u == v){
return (ans + queryS(id[u],id[v],1,n,1));
}
else{
if(dep[u] > dep[v]){
swap(u,v);
}
return (ans + queryS(id[u],id[v],1,n,1));
}
}
int FindM(int u,int v){
int ans = -INF;
while(top[u] != top[v]){ //不在同一条链上,就从u所在转移到v所在
if(dep[top[u]] < dep[top[v]]){
swap(u,v);
}
ans = max(ans,queryM(id[top[u]],id[u],1,n,1));
u = fa[top[u]];
}
if(u == v){
return max(ans,queryM(id[u],id[v],1,n,1));
}
else{
if(dep[u] > dep[v]){
swap(u,v);
}
return max(ans,queryM(id[u],id[v],1,n,1));
}
}
///////////////////////////////////////////////////
void init() {
tol = cnt = 0;
memset(fa,0,sizeof fa);
memset(siz,0,sizeof siz);
memset(son,0,sizeof son);
memset(dep,0,sizeof dep);
memset(top,0,sizeof top);
memset(id,0,sizeof id);
memset(sum,0,sizeof sum);
memset(node,0,sizeof node);
memset(mx,0,sizeof mx);
memset(head,-1,sizeof head);
}
int main(){
int q;
while(~scanf("%d",&n)){
int u,v;
init();
for(int i = 1;i < n;i++){
scanf("%d%d",&u,&v);
addnode(u,v);
addnode(v,u);
}
dfs1(1,0,1);
dfs2(1,1);
for(int i = 1;i <= n;i++){
int w;
scanf("%d",&w);
update(id[i],w,1,n,1);
}
scanf("%d",&q);
char s[20];
while(q--){
scanf("%s%d%d",s,&u,&v);
if(s[0] == 'C'){
update(id[u],v,1,n,1);
}
else if(s[1] == 'S'){
printf("%d\n",FindS(u,v));
}
else{
printf("%d\n",FindM(u,v));
}
}
}
return 0;
}
J
HYSBZ 1036 树的统计Count(树链剖分)题解的更多相关文章
- 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 树链剖分+线段树
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 (树链剖分)(线段树单点修改)
[ZJOI2008]树的统计Count Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 14968 Solved: 6079[Submit][Stat ...
- 【BZOJ1036】[ZJOI2008]树的统计Count 树链剖分
[BZOJ1036][ZJOI2008]树的统计Count Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成一些操作: I. ...
- bzoj1036 [ZJOI2008]树的统计Count 树链剖分模板题
[ZJOI2008]树的统计Count Description 一棵树上有n个节点,编号分别为1到n,每个节点都有一个权值w.我们将以下面的形式来要求你对这棵树完成 一些操作: I. CHANGE u ...
- Cogs 1688. [ZJOI2008]树的统计Count(树链剖分+线段树||LCT)
[ZJOI2008]树的统计Count ★★★ 输入文件:bzoj_1036.in 输出文件:bzoj_1036.out 简单对比 时间限制:5 s 内存限制:162 MB [题目描述] 一棵树上有n ...
- BZOJ 1036 [ZJOI2008]树的统计Count (树链剖分 - 点权剖分 - 单点权修改)
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1036 树链剖分模版题,打的时候注意点就行.做这题的时候,真的傻了,单词拼错检查了一个多小时 ...
随机推荐
- 【Java nio】 NonBlocking NIO
package com.slp.nio; import org.junit.Test; import java.io.IOException; import java.net.InetSocketAd ...
- VScode之JavaScript Snippet Pack
一个片段包 使用例如: cl 回车或者tab键,就可以完整的打出console.log("") 还有很多快捷功能: 参考: https://marketplace.visualst ...
- vscode新建html,没有模板
首先,在文件夹下右击--新建--index.html 输入! 按tab键 完成!
- iOS UITextField输入后隐藏键盘
1.首先在Interface Builder中选择TextFields,然后在Text Field Attributes中找到Text Input Traits,选择Return Key为done. ...
- 获得当前正在显示的activity的类名
需要加一个权限: <uses-permission android:name="android.permission.GET_TASKS"/> ActivityMana ...
- 微信小程序 --- if/else条件渲染
if 条件渲染:当为真的时候显示,当为假的时候隐藏: else 条件渲染:当为真的时候隐藏,当为假的时候显示: <view wx:if="{{true}}">{{tex ...
- [面经] 南京SAP面试(下)
上一篇讲到了一面结束,这一篇说说剩下的事情. 周三上午一面完了之后回去上班,本以为要等几天才会二面,结果那个经理M下午就打电话给我,约了第二天(周四)下午过去面试,会有Boss从上海过来面,办事效率还 ...
- 160227、javascript特效
1.给网页设定快捷键 js: function getkey(){ event = event || window.event; url = "www.baidu.com&q ...
- Oracle归档文件夹权限设置错误导致的数据库问题解决
把oracle设置为归档模式并且为归档文件新建文件夹 /home/oracle/app/oracle/arch/orcl 但是在启动或者备份时候经常性出现错误 startup报错 startup同时日 ...
- With all Java versions it is strongly recommended to not use experimental -XX JVM options.
https://lucene.apache.org/solr/7_6_0//SYSTEM_REQUIREMENTS.html System Requirements Apache Solr runs ...