QTREE 树链剖分---模板 spoj QTREE
《树链剖分及其应用》 一文讲得非常清楚,我一早上就把他学会了并且A了这题的入门题。
spoj QTREE
题目:
给出一棵树,有两种操作:
1.修改一条边的边权。
2.询问节点a到b的最大边权。
直接粘代码。更成熟的代码可以看下一篇BZOJ 1036: [ZJOI2008]树的统计Count
#include <set>
#include <map>
#include <list>
#include <cmath>
#include <queue>
#include <stack>
#include <string>
#include <vector>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm> using namespace std; typedef long long ll;
typedef unsigned long long ull; #define debug puts("here")
#define rep(i,n) for(int i=0;i<n;i++)
#define rep1(i,n) for(int i=1;i<=n;i++)
#define REP(i,a,b) for(int i=a;i<=b;i++)
#define foreach(i,vec) for(unsigned i=0;i<vec.size();i++)
#define pb push_back
#define RD(n) scanf("%d",&n)
#define RD2(x,y) scanf("%d%d",&x,&y)
#define RD3(x,y,z) scanf("%d%d%d",&x,&y,&z)
#define RD4(x,y,z,w) scanf("%d%d%d%d",&x,&y,&z,&w)
#define All(vec) vec.begin(),vec.end()
#define MP make_pair
#define PII pair<int,int>
#define PQ priority_queue
#define cmax(x,y) x = max(x,y)
#define cmin(x,y) x = min(x,y)
#define Clear(x) memset(x,0,sizeof(x))
/* #pragma comment(linker, "/STACK:1024000000,1024000000") int size = 256 << 20; // 256MB
char *p = (char*)malloc(size) + size;
__asm__("movl %0, %%esp\n" :: "r"(p) ); */ /******** program ********************/ const int MAXN = ; int val[MAXN];
int po[MAXN],tol;
bool use[MAXN];
int n; struct node{
int y,c,next;
}edge[MAXN]; inline void add(int x,int y,int c){
edge[++tol].y = y;
edge[tol].c = c;
edge[tol].next = po[x];
po[x] = tol;
} struct tc{ // tree chain subdivision
int sz; // x子树大小
int dep; // 节点x的深度
int top; // 节点x所在链的顶端节点
int fa; // 节点x的父亲
int son; // 重儿子
int tid; // 在线段树中的位置
}p[MAXN]; void dfsFind(int x,int fa,int dep){
use[x] = true;
p[x].dep = dep;
p[x].sz = ;
p[x].son = ;
p[x].fa = fa; int mx = ;
for(int i=po[x];i;i=edge[i].next){
int y = edge[i].y;
if(use[y])continue;
dfsFind(y,x,dep+);
p[x].sz += p[y].sz;
if(p[y].sz>mx){
p[x].son = y;
mx = p[y].sz;
}
}
} int tid;
void dfsCon(int x,int fa){
use[x] = true;
p[x].tid = ++ tid;
p[x].top = fa;
if(p[x].son)
dfsCon(p[x].son,fa);
for(int i=po[x];i;i=edge[i].next){
int y = edge[i].y;
if(use[y])continue;
dfsCon(y,y);
}
} struct Tree{
int l,r,mx;
inline int mid(){
return (l+r)>>;
}
}tree[MAXN<<]; inline void update(int rt){
tree[rt].mx = max(tree[rt<<].mx,tree[rt<<|].mx);
} void build(int l,int r,int rt){
tree[rt].l = l;
tree[rt].r = r;
if(l==r){
tree[rt].mx = val[l];
return;
}
int mid = tree[rt].mid();
build(l,mid,rt<<);
build(mid+,r,rt<<|); update(rt);
} void modify(int pos,int c,int rt){
if(tree[rt].l==tree[rt].r){
tree[rt].mx = c;
return;
}
int mid = tree[rt].mid();
if(pos<=mid)
modify(pos,c,rt<<);
else
modify(pos,c,rt<<|); update(rt);
} int ask(int l,int r,int rt){
if(tree[rt].l==l&&tree[rt].r==r)
return tree[rt].mx;
int mid = tree[rt].mid();
if(r<=mid)
return ask(l,r,rt<<);
else if(l>mid)
return ask(l,r,rt<<|);
else
return max( ask(l,mid,rt<<),ask(mid+,r,rt<<|) );
} int main(){ #ifndef ONLINE_JUDGE
freopen("sum.in","r",stdin);
//freopen("sum.out","w",stdout);
#endif int x,y,z,ncase;
RD(ncase);
while(ncase--){
RD(n);
Clear(po);
tol = ; REP(i,,n){
RD3(x,y,z);
add(x,y,z);
add(y,x,z);
} Clear(use);
dfsFind(,,); tid = ;
Clear(use);
dfsCon(,); for(int i=;i<tol;i+=){
int x = edge[i^].y; // 对应于第x条边的节点 (x,y)
int y = edge[i].y; if(p[x].dep>p[y].dep)
val[ p[x].tid ] = edge[i].c;
else
val[ p[y].tid ] = edge[i].c;
} build(,n,); char op[];
while(scanf("%s",op),op[]!='D'){ if(op[]=='C'){
RD2(x,z);
y = edge[x<<].y;
x = edge[x<<|].y; if( p[x].dep>p[y].dep )
modify( p[x].tid,z, );
else
modify( p[y].tid,z, );
}else{
RD2(x,y);
int ans = -(<<);
while( p[x].top != p[y].top ){
if( p[ p[x].top ].dep < p[ p[y].top ].dep )
swap(x,y);
ans = max(ans,ask(p[ p[x].top ].tid,p[x].tid,));
x = p[ p[x].top ].fa;
}
if(p[x].dep>p[y].dep)
swap(x,y);
if(x!=y)
ans = max(ans,ask(p[x].tid+,p[y].tid,));
printf("%d\n",ans);
}
}
} return ;
}
QTREE 树链剖分---模板 spoj QTREE的更多相关文章
- BZOJ 2243 染色 | 树链剖分模板题进阶版
BZOJ 2243 染色 | 树链剖分模板题进阶版 这道题呢~就是个带区间修改的树链剖分~ 如何区间修改?跟树链剖分的区间询问一个道理,再加上线段树的区间修改就好了. 这道题要注意的是,无论是线段树上 ...
- 算法复习——树链剖分模板(bzoj1036)
题目: 题目背景 ZJOI2008 DAY1 T4 题目描述 一棵树上有 n 个节点,编号分别为 1 到 n ,每个节点都有一个权值 w .我们将以下面的形式来要求你对这棵树完成一些操作:I.CHAN ...
- Hdu 5274 Dylans loves tree (树链剖分模板)
Hdu 5274 Dylans loves tree (树链剖分模板) 题目传送门 #include <queue> #include <cmath> #include < ...
- SPOJ QTREE - Query on a tree 【树链剖分模板】
题目链接 引用到的大佬博客 代码来自:http://blog.csdn.net/jinglinxiao/article/details/72940746 具体算法讲解来自:http://blog.si ...
- SPOJ QTREE Query on a Tree【树链剖分模板题】
树链剖分,线段树维护~ #include <cstdio> #include <cstring> #include <iostream> #include < ...
- SPOJ 375 Query on a tree(树链剖分)(QTREE)
You are given a tree (an acyclic undirected connected graph) with N nodes, and edges numbered 1, 2, ...
- SPOJ QTREE 树链剖分
树链剖分的第一题,易懂,注意这里是边. #include<queue> #include<stack> #include<cmath> #include<cs ...
- Spoj Query on a tree SPOJ - QTREE(树链剖分+线段树)
You are given a tree (an acyclic undirected connected graph) with N nodes, and edges numbered 1, 2, ...
- SPOJ375.QTREE树链剖分
题意:一个树,a b c 代表a--b边的权值为c.CHANGE x y 把输入的第x条边的权值改为y,QUERY x y 查询x--y路径上边的权值的最大值. 第一次写树链剖分,其实树链剖分只能说 ...
随机推荐
- Redis安装与调试
Redis安装与调试 Redis安装与调试linux版本:64位CentOS 6.5 Redis版本:2.8.17 (更新到2014年10月31日) Redis官网:http://redis.io/ ...
- GUI图形界面
一.界面元素 1.首层容器:Jwindow.JFrane(窗体).JDialge(对话框).JApplet. 2.组件:标签.文本框.密码框.按钮.下拉列表.单选框.复选框.文本域等 注:标签JLab ...
- VMware的“桥接”、“NAT”、“Host-only”上网方式的区别
http://liblog.littleyuan.com/archives/9 在说到VMware的网络模型之前,先说一下VMware的几个虚拟设备: VMnet0:这是VMware用于虚拟桥接网络下 ...
- 测试URL有效性
方法一: #禁用滚动条 $ProgressPreference='silentlycontinue' Invoke-WebRequest "www.163.com" -UseBas ...
- Codeforces Beta Round #51 A. Flea travel 水题
A. Flea travel Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/55/problem ...
- makefile中的patsubst
函数名称:加前缀函数—addprefix. 函数功能:为“NAMES…”中的每个文件名称加入前缀“PREFIX”.參数“NAMES…”是空格切割的文件名称序列,将“SUFFIX”加入到此序列的每个文件 ...
- Swift 3.0 使用Core Data
swift版本:3.0 Xcode版本:8.0 iOS版本:10.0 自iOS10 和swift3.0 之后,苹果的访问CoreData的方法发生了很大改变,简洁了许多,下面的内容是从0开始建立一个e ...
- 日志分析(五) PV&UV
应用设计请求之初,对于url有一定的规划.因此,请求的url格式类似如下: /**/school/****?token=a66cb2a3-e0b7-4f0a-b332-********* token唯 ...
- 不安装oracle客户端也可以使用pl/sql developer
通常情况下,用PL/SQL Developer连接Oracle是需要安装Oracle客户端软件的,这也就意味着你的硬盘将被占用大约1G-2G的空间,对于Windows操作系统来说,你还会多出一些开机自 ...
- Discuz!X/模板标签说明
Discuz 模板标签说明 Discuz! 的模板采用近似 PHP 表达式的语法,基本都是可识别的HTML,但涉及到变量和动态内容时,基本形式下: <!-{ 代码内容 }-> 逻辑元素包围 ...