《树链剖分及其应用》 一文讲得非常清楚,我一早上就把他学会了并且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的更多相关文章

  1. BZOJ 2243 染色 | 树链剖分模板题进阶版

    BZOJ 2243 染色 | 树链剖分模板题进阶版 这道题呢~就是个带区间修改的树链剖分~ 如何区间修改?跟树链剖分的区间询问一个道理,再加上线段树的区间修改就好了. 这道题要注意的是,无论是线段树上 ...

  2. 算法复习——树链剖分模板(bzoj1036)

    题目: 题目背景 ZJOI2008 DAY1 T4 题目描述 一棵树上有 n 个节点,编号分别为 1 到 n ,每个节点都有一个权值 w .我们将以下面的形式来要求你对这棵树完成一些操作:I.CHAN ...

  3. Hdu 5274 Dylans loves tree (树链剖分模板)

    Hdu 5274 Dylans loves tree (树链剖分模板) 题目传送门 #include <queue> #include <cmath> #include < ...

  4. SPOJ QTREE - Query on a tree 【树链剖分模板】

    题目链接 引用到的大佬博客 代码来自:http://blog.csdn.net/jinglinxiao/article/details/72940746 具体算法讲解来自:http://blog.si ...

  5. SPOJ QTREE Query on a Tree【树链剖分模板题】

    树链剖分,线段树维护~ #include <cstdio> #include <cstring> #include <iostream> #include < ...

  6. 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, ...

  7. SPOJ QTREE 树链剖分

    树链剖分的第一题,易懂,注意这里是边. #include<queue> #include<stack> #include<cmath> #include<cs ...

  8. 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, ...

  9. SPOJ375.QTREE树链剖分

    题意:一个树,a b c 代表a--b边的权值为c.CHANGE x y  把输入的第x条边的权值改为y,QUERY x y 查询x--y路径上边的权值的最大值. 第一次写树链剖分,其实树链剖分只能说 ...

随机推荐

  1. 【转】C#传委托给C的函数指针调用问题

    C#传委托给C的函数指针调用问题C代码如下: #include "stdio.h" __declspec(dllexport) int Call(int (*qq)(int num ...

  2. CodeForces 706D Vasiliy's Multiset (字典树查询+贪心)

    题意:最开始的时候有一个集合,集合里面只有一个元素0,现在有q次操作,操作分为3种: + x: 表示向集合中添加一个元素x - x:表示删除集合中值为x的一个元素 ? x:表示查询集合中与x异或的最大 ...

  3. [html]html常用代码

    上传文件表单属性 enctype="multipart/form-data" 单选(是否选中) checked="checked" 下拉列表(是否选中) sel ...

  4. The plot Function in matlab

    from http://pundit.pratt.duke.edu/wiki/MATLAB:Plotting The plot Function The plot function is used t ...

  5. js字符串常用判断方法

    转自:http://blog.sina.com.cn/s/blog_6819fa800100j5t6.html 一.方法介绍 function obj$(id)                     ...

  6. rxjava各种使用场景

    1. 数据的三级缓存 final Observable memory = Observable.create(new Observable.OnSubscribe() { @Override publ ...

  7. TChromeTabs 优化改进

    已知未解决问题 全屏时当窗体失去焦点,则会显示出未绘制完成的原标题栏(Fixed): 处于非 Areo 效果下时,窗体标题栏需要定制. 新增按钮上的 Hint 提示后再移至其它标签,将无法重新提示. ...

  8. Codeforces Round #Pi (Div. 2) E. President and Roads tarjan+最短路

    E. President and RoadsTime Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/567 ...

  9. Android数据的四种存储方式之SQLite数据库

    Test.java: /** * 本例解决的问题: * 核心问题:通过SQLiteOpenHelper类创建数据库对象 * 通过数据库对象对数据库的数据的操作 * 1.sql语句方式操作SQLite数 ...

  10. Python学习 之 运算符&表达式

    1.Python运算符包括:赋值运算符.算术运算符.关系运算符.逻辑运算符. 表达式是将不同的数据(包括变量.函数)用运算符号按一定规则连接起来的一种式子. 2.赋值运算符:=.+=.-=.*=./= ...