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路径上边的权值的最大值. 第一次写树链剖分,其实树链剖分只能说 ...
随机推荐
- codeforces 630 I(规律&&组合)
I - Parking Lot Time Limit:500MS Memory Limit:65536KB 64bit IO Format:%I64d & %I64u Subm ...
- [MAC OSX - 1] OSX10.10不能安装JKD8,不能使用eclipse
(1)电脑升级为10.10后,打开eclipse总是提示"您需要安装旧 Java SE 6 运行环境才能打开"Eclipse". 解决:安装JKD (2)不能安装JK ...
- Android:从程序员到架构师之路Ⅲ_高焕堂
Part-2: 从Android框架代码中学习设计 一 基础设计模式(Pattern)的代码:以Android为例 1.Template Method模式:IoC(控制反转)机制 2.Observer ...
- Codeforces Round #257 (Div. 1) C. Jzzhu and Apples (素数筛)
题目链接:http://codeforces.com/problemset/problem/449/C 给你n个数,从1到n.然后从这些数中挑选出不互质的数对最多有多少对. 先是素数筛,显然2的倍数的 ...
- HDU 4432 Sum of divisors (水题,进制转换)
题意:给定 n,m,把 n 的所有因数转 m 进制,再把各都平方,求和. 析:按它的要求做就好,注意的是,是因数,不可能有重复的...比如4的因数只有一个2,还有就是输出10进制以上的,要用AB.. ...
- Spring Data JPA教程, 第八部分:Adding Functionality to a Repository (未翻译)
The previous part of my tutorial described how you can paginate query results with Spring Data JPA. ...
- Random的nextInt用法
因为想当然的认为Random类中nextInt()(注:不带参数),会产生伪随机的正整数,采用如下的方式生成0~99之间的随机数: Random random = new Random(); rand ...
- C#学习笔记(十四):GC机制和弱引用
垃圾回收(GC) 垃圾回收即Garbage Collector,垃圾指的是内存中已经不会再使用的对象,通过收集释放掉这些对象占用的内存. GC以应用程序的root为基础,遍历应用程序在Heap上动态分 ...
- 访问修饰符与可选修饰符static的使用
在Java中修饰符较多,在这里简单说一说几个访问修饰符的选择使用,和可选修饰符static的使用. 一.访问修饰符:这是Java里用来控制访问类及类的方法.变量的访问权限,从而达到只暴露接口,来隐藏内 ...
- mount nfs的可选参数
mount nfs的可选参数:HARD mount和SOFT MOUNT:HARD:NFS CLIENT会不断的尝试与SERVER的连接(在后台,不会给出任何提示信息,在LINUX下有的版本仍然会给出 ...