SPOJ OTOCI

裸的动态树问题。

回顾一下我们对树的认识。

最初,它是一个连通的无向的无环的图,然后我们发现由一个根出发进行BFS 会出现层次分明的树状图形。

然后根据树的递归和层次性质,我们得到了很多有趣的算法,比如单源最短路等等。

如今,我们面对更复杂的问题,给定一个森林,随时更改树的形态,并询问两个节点之间路径上所有点的权值和。

要求复杂度O(Logn)

我们把树看作由若干条链构成,每个链用一个splay维护。

至此,我们对树的认识已经到了一个比较高的水平了。

关于LCT的详细阐述,参考http://www.cnblogs.com/zinthos/p/3900225.html

代码并不复杂, 一句话 splay的核心是splay(x) LCT的核心是access(x)

此处join(x,y)通过对x的翻转实现,splay(x)后,反转x x就由队尾变成队首此时PNT(x)==null, PNT(x)=y就实现了连接(x,y).

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm> using namespace std; const int MaxNode=31000; int Lch[MaxNode];
int Rch[MaxNode];
int Pnt[MaxNode];
int Data[MaxNode];
int Sum[MaxNode];
int Rev[MaxNode];
int List[MaxNode];
int Total; inline bool isRoot(int t){
return (!Pnt[t]||(Lch[Pnt[t]]!=t&&Rch[Pnt[t]]!=t));
}
inline void Update(int cur){
Sum[cur]=Sum[Lch[cur]]+Sum[Rch[cur]]+Data[cur];
}
void Reverse(int cur){
if (!Rev[cur]) return;
swap(Lch[cur],Rch[cur]);
Rev[Lch[cur]]^=1;
Rev[Rch[cur]]^=1;
Rev[cur]=0;
}
void LeftRotate(int cur){
if (isRoot(cur)) return;
int pnt=Pnt[cur],anc=Pnt[pnt];
Lch[pnt]=Rch[cur];
if (Rch[cur]) Pnt[Rch[cur]]=pnt;
Rch[cur]=pnt;
Pnt[pnt]=cur;
Pnt[cur]=anc;
if (anc){
if (Lch[anc]==pnt) Lch[anc]=cur;
else if (Rch[anc]==pnt) Rch[anc]=cur;
}
Update(pnt);
Update(cur);
}
void RightRotate(int cur){
if (isRoot(cur)) return;
int pnt=Pnt[cur],anc=Pnt[pnt];
Rch[pnt]=Lch[cur];
if (Lch[cur]) Pnt[Lch[cur]]=pnt;
Lch[cur]=pnt;
Pnt[pnt]=cur;
Pnt[cur]=anc;
if (anc){
if (Rch[anc]==pnt) Rch[anc]=cur;
else if (Lch[anc]==pnt) Lch[anc]=cur;
}
Update(pnt);
Update(cur);
}
void Splay(int cur){
int pnt,anc;
List[++Total]=cur;
for (int i=cur;!isRoot(i);i=Pnt[i]) List[++Total]=Pnt[i];
for (;Total;--Total)
if (Rev[List[Total]]) Reverse(List[Total]);
while (!isRoot(cur)){
pnt=Pnt[cur];
if (isRoot(pnt)){// 父亲是根结点,做一次旋转
if (Lch[pnt]==cur) LeftRotate(cur);
else RightRotate(cur);
}
else{
anc=Pnt[pnt];
if (Lch[anc]==pnt){
if (Lch[pnt]==cur) LeftRotate(pnt),LeftRotate(cur);// 一条线
else RightRotate(cur),LeftRotate(cur);// 相反两次
}
else{
if (Rch[pnt]==cur) RightRotate(pnt),RightRotate(cur);// 一条线
else LeftRotate(cur),RightRotate(cur);// 相反两次
}
}
}
}
int Expose(int u){
int v=0;
for (;u;u=Pnt[u]) Splay(u),Rch[u]=v,v=u,Update(u);
for (;Lch[v];v=Lch[v]);
return v;
}
void Modify(int x,int d){
Splay(x);
Data[x]=d;
Update(x);
}
int Query(int x,int y){
int rx=Expose(x),ry=Expose(y);
if (rx==ry){
for (int u=x,v=0;u;u=Pnt[u]){
Splay(u);
if (!Pnt[u]) return Sum[Rch[u]]+Data[u]+Sum[v];
Rch[u]=v;
Update(u);
v=u;
}
}
return -1;
}
bool Join(int x,int y){
int rx=Expose(x),ry=Expose(y);
if (rx==ry) return false;
else{
Splay(x);
Rch[x]=0;
Rev[x]=1;
Pnt[x]=y;
Update(x);
return true;
}
}
void Cut(int x){
if (Pnt[x]){
Expose(x);
Pnt[Lch[x]]=0;
Lch[x]=0;
Update(x);
}
}
int n,Q; void init(){
Total=0;
memset(Rev,0,sizeof(Rev));
memset(Pnt,0,sizeof(Pnt));
memset(Lch,0,sizeof(Lch));
memset(Rch,0,sizeof(Rch));
memset(Sum,0,sizeof(Sum));
}
char cmd[22];
int main()
{
init();
scanf("%d",&n);
for (int i=1;i<=n;i++) scanf("%d",&Data[i]);
scanf("%d",&Q);
while (Q--){
int x,y;
scanf("%s%d%d",cmd,&x,&y);
if (cmd[0]=='p'){
Modify(x,y);
}
if (cmd[0]=='b'){
if (Join(x,y)) printf("yes\n");
else printf("no\n");
}
if (cmd[0]=='e'){
int ans=Query(x,y);
if (ans==-1) printf("impossible\n");
else printf("%d\n",ans);
}
}
return 0;
}

  

SPOJ OTOCI 动态树 LCT的更多相关文章

  1. hdu 5398 动态树LCT

    GCD Tree Time Limit: 5000/2500 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Su ...

  2. hdu 5002 (动态树lct)

    Tree Time Limit: 16000/8000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Submi ...

  3. 动态树LCT小结

    最开始看动态树不知道找了多少资料,总感觉不能完全理解.但其实理解了就是那么一回事...动态树在某种意思上来说跟树链剖分很相似,都是为了解决序列问题,树链剖分由于树的形态是不变的,所以可以通过预处理节点 ...

  4. bzoj2049-洞穴勘测(动态树lct模板题)

    Description 辉辉热衷于洞穴勘测.某天,他按照地图来到了一片被标记为JSZX的洞穴群地区.经过初步勘测,辉辉发现这片区域由n个洞穴(分别编号为1到n)以及若干通道组成,并且每条通道连接了恰好 ...

  5. [模板] 动态树/LCT

    简介 LCT是一种数据结构, 可以维护树的动态加边, 删边, 维护链上信息(满足结合律), 单次操作时间复杂度 \(O(\log n)\).(不会证) 思想类似树链剖分, 因为splay可以换根, 用 ...

  6. 动态树LCT(Link-cut-tree)总结+模板题+各种题目

    一.理解LCT的工作原理 先看一道例题: 让你维护一棵给定的树,需要支持下面两种操作: Change x val:  令x点的点权变为val Query x y:  计算x,y之间的唯一的最短路径的点 ...

  7. HDU 4718 The LCIS on the Tree (动态树LCT)

    The LCIS on the Tree Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 65535/65535 K (Java/Oth ...

  8. BZOJ 2002: [Hnoi2010]Bounce 弹飞绵羊 (动态树LCT)

    2002: [Hnoi2010]Bounce 弹飞绵羊 Time Limit: 10 Sec  Memory Limit: 259 MBSubmit: 2843  Solved: 1519[Submi ...

  9. HDU 5002 Tree(动态树LCT)(2014 ACM/ICPC Asia Regional Anshan Online)

    Problem Description You are given a tree with N nodes which are numbered by integers 1..N. Each node ...

随机推荐

  1. centos6 用户登陆管理

    查看当前登陆有哪些用户,在做什么 [root@web01 ~]# w :: up :, users, load average: 0.00, 0.00, 0.00 USER TTY FROM LOGI ...

  2. Volume 1. Sorting/Searching(uva)

    340 - Master-Mind Hints /*读了老半天才把题读懂,读懂了题输出格式没注意,结果re了两次. 题意:先给一串数字S,然后每次给出对应相同数目的的一串数字Si,然后优先统计Si和S ...

  3. Django之Ajax提交

    Ajax 提交数据,页面不刷新 Ajax要引入jQuery Django之Ajax提交 Js实现页面的跳转: location.href = "/url/" $ajax({ url ...

  4. python 库文件版本收集及安装

    版本收集:pip freeze > require.txt版本安装:pip install -r require.txt

  5. 窗口类WNDCLASSEX名词解析

    窗口类WNDCLASSEX名词解析 typedef struct tagWNDCLASSEX{ UINT cbsize; UINT style; WNDPROC lpfnWNDProc; int cb ...

  6. wps填充1到1000

    A1单元格1 ,选中,填充,序列,确定

  7. Leetcode 137.只出现一次的数字II

    只出现一次的数字II 给定一个非空整数数组,除了某个元素只出现一次以外,其余每个元素均出现了三次.找出那个只出现了一次的元素. 说明: 你的算法应该具有线性时间复杂度. 你可以不使用额外空间来实现吗? ...

  8. HTML5 & CSS3 & font-family

    HTML5 & CSS3 & font-family 中文字体的英文名称 宋体* SimSun 黑体* SimHei 微软雅黑* Microsoft YaHei 微软正黑体 http: ...

  9. 【转】Java的path,classpath,java_home环境变量的配置与具体含义

    对于一个Java初学者来说,第一步要做的是安装jdk并配置环境变量,一般按照书上或者网上的步骤,一步步照着做就行了,但是对于初学者来说,很多问题没有解决,比如为什么很多配置方法各不相同,却都能够配置成 ...

  10. 【BZOJ1758】重建计划(点分治)

    题意: 给定一棵n个点的树,每条边有权值.求一条链,这条链包含的边数在L和U之间,且平均边权最大.N﹤=100000 思路:RYZ作业 二分答案再点分治,寻找是否有大于0且边数在L和U之间的链 f[i ...