之前做了两道点权的LCT,这次做一下边权的LCT.上网找了一下资料,发现对于边权的LCT有这么两种处理方法,一种是每条边建一个点,于是边权就转成点权了。另外一种则是每个边权对应到点权上,也就是每个点对应它的父边,这就要求树不能换根,不换根提路径是有点蛋疼的,所以就需要知道怎么在不换根的时候提取出u到 v的路径,实现的方法是基于expose的一种lca。

传统的expose是一直expose到根,另一种做法则是,对于u,v,expose(v),splay(v),这样v到父亲的边打成了重链,然后相应的执行一种类似于expose(u)的操作,但是中止条件改为当u->fa==null的时候就应该结束while,因为u->fa==null说明u和v在同一条重链上,或者说此时u就是lca(u,v),利用lca函数(传指针的引用),最后v和u->ch[1]就是lca到u,v的两个方向的路径。具体可以对比一下expose和lca函数

另外一点是我在debug的时候学习到的,原来的模板里的expose是没有由下往上更新的,也就是说expose(v)之后,v上的所有祖先的信息都是没有更新的,因此expose之后还要splay(v),将信息往上带,但是由于lca函数传的是引用,调用完之后原本的信息就丢失掉了,所以可以在循环里直接upd,把正确的信息往上传。

时间4600ms,差点超时,做这题其实也只是为了学习如何用LCT来操作边权。

#pragma warning(disable:4996)
#include <iostream>
#include <cstring>
#include <string>
#include <vector>
#include <cstdio>
#include <algorithm>
#include <cmath>
using namespace std; #define ll long long
#define maxn 12000
#define NINF -1000000000 struct Edge
{
int u, v, w;
Edge(int ui, int vi, int wi) :u(ui), v(vi), w(wi){}
Edge(){}
}E[maxn * 2], etop; struct Node
{
Node *p, *ch[2];
int val;
int mx;
int size;
bool isRoot;
Node *fa;
Node(){
val = NINF;
size = 0; isRoot = false;
mx = NINF;
}
void setc(Node *c, int d){
ch[d] = c;
c->p = this;
}
bool d(){
return p->ch[1] == this;
}
void upd(){
mx = max(val, max(ch[0]->mx, ch[1]->mx));
}
void relax();
void setRoot(Node *f);
}Tnull, *null = &Tnull; void Node::relax(){ } void Node::setRoot(Node *f){
isRoot = true;
fa = f; p = null;
} Node mem[maxn], *C = mem; Node *make(int v){
C->val = C->mx = v;
C->ch[0] = C->ch[1] = null;
C->isRoot = true;
C->p = null;
C->fa = null;
return C++;
} void rot(Node *t){
Node *p = t->p;
p->relax();
t->relax();
bool d = t->d();
p->p->setc(t, p->d());
p->setc(t->ch[!d], d);
t->setc(p, !d);
p->upd();
if (p->isRoot){
p->isRoot = false;
t->isRoot = true;
t->fa = p->fa;
}
} void pushTo(){ } void splay(Node *u, Node *f = null){
pushTo();
while (u->p != f){
if (u->p->p == f) rot(u);
else u->d() == u->p->d() ? (rot(u->p), rot(u)) : (rot(u), rot(u));
}
u->upd();
} Node *v[maxn];
vector<int> G[maxn];
vector<int> W[maxn];
int n; int nQ; int que[maxn], fa[maxn], qh = 0, qt = 0;
int dep[maxn];
int wht[maxn]; void bfs(){
qh = qt = 0;
que[qt++] = 1;
fa[1] = -1;
while (qh < qt){
int u = que[qh++];
for (int i = 0; i < G[u].size(); i++){
int e = G[u][i];
if (e != fa[u]){
fa[e] = u; v[e]->fa = v[u]; que[qt++] = e;
}
}
}
} Node *expose(Node *u){
Node *v;
for (v = null; u != null; v = u, u = u->fa){
splay(u);
u->ch[1]->setRoot(u);
u->setc(v, 1);
}
return v;
} void dfs(int u, int fa)
{
for (int i = 0; i < G[u].size(); i++){
int v = G[u][i], w = W[u][i];
if (v == fa) continue;
dep[v] = dep[u] + 1;
wht[v] = w;
dfs(v, u);
}
} void lca(Node *&u, Node *&v){
expose(v); splay(v);
for (v = null; u != null; v = u, u = u->fa){
splay(u);
if (u->fa == null) return;
u->ch[1]->setRoot(u);
u->setc(v, 1);
u->upd();
}
} int main()
{
int T; cin >> T;
while (T--)
{
scanf("%d", &n);
memset(dep, 0, sizeof(dep));
for (int i = 0; i <= n; i++) G[i].clear(), W[i].clear();
for (int i = 0; i < n - 1; i++){
scanf("%d%d%d", &E[i].u, &E[i].v, &E[i].w);
G[E[i].u].push_back(E[i].v);
G[E[i].v].push_back(E[i].u);
W[E[i].u].push_back(E[i].w);
W[E[i].v].push_back(E[i].w);
}
dep[1] = 1;
dfs(1, -1); wht[1] = NINF;
C = mem;
for (int i = 1; i <= n; i++){
v[i] = make(wht[i]);
}
bfs();
char cmd[10]; int ai, bi;
while (scanf("%s", cmd)){
if (cmd[0] == 'D') break;
scanf("%d%d", &ai, &bi);
if (cmd[0] == 'Q'){
Node *ui = v[ai], *vi = v[bi];
lca(ui, vi);
printf("%d\n", max(ui->ch[1]->mx, vi->mx));
}
else{
--ai;
int ui = E[ai].u, vi = E[ai].v;
if (dep[ui] < dep[vi]) swap(ui, vi);
Node *u = v[ui];
expose(u);
splay(u);
u->val = bi;
splay(u);
}
}
}
return 0;
}

SPOJ375 Query on a tree(LCT边权)的更多相关文章

  1. SPOJ375 Query on a tree

    Description You are given a tree (an acyclic undirected connected graph) with N nodes, and edges num ...

  2. spoj 375 query on a tree LCT

    这道题是树链剖分的裸题,正在学LCT,用LCT写了,发现LCT代码比树链剖分还短点(但我的LCT跑极限数据用的时间大概是kuangbin大神的树链剖分的1.6倍,所以在spoj上是850ms卡过的). ...

  3. SPOJ375 Query on a tree(树链剖分)

    传送门 题意 给出一棵树,每条边都有权值,有两种操作: 把第p条边的权值改为x 询问x,y路径上的权值最大的边 code #include<cstdio> #include<algo ...

  4. SPOJ375 Query on a tree 【倍增,在线】

    题目链接[http://www.spoj.com/problems/QTREE/] 题意:给出一个包含N(N<=10000)节点的无根树,有多次询问,询问的方式有两种1.DIST  a b 求a ...

  5. QTREE5 - Query on a tree V——LCT

    QTREE5 - Query on a tree V 动态点分治和动态边分治用Qtree4的做法即可. LCT: 换根后,求子树最浅的白点深度. 但是也可以不换根.类似平常换根的往上g,往下f的拼凑 ...

  6. SPOJ 375. Query on a tree (动态树)

    375. Query on a tree Problem code: QTREE You are given a tree (an acyclic undirected connected graph ...

  7. QTREE6 - Query on a tree VI 解题报告

    QTREE6 - Query on a tree VI 题目描述 给你一棵\(n\)个点的树,编号\(1\)~\(n\).每个点可以是黑色,可以是白色.初始时所有点都是黑色.下面有两种操作请你操作给我 ...

  8. Query on a tree——树链剖分整理

    树链剖分整理 树链剖分就是把树拆成一系列链,然后用数据结构对链进行维护. 通常的剖分方法是轻重链剖分,所谓轻重链就是对于节点u的所有子结点v,size[v]最大的v与u的边是重边,其它边是轻边,其中s ...

  9. 动态树(Link Cut Tree) :SPOJ 375 Query on a tree

    QTREE - Query on a tree #number-theory You are given a tree (an acyclic undirected connected graph) ...

随机推荐

  1. Orcle 系统表

    oracle系统表大全 一.管理员 1.用户: select username from dba_users; 改口令 alter user spgroup identified by spgtest ...

  2. About Inside the Azure Storage Outage of November 18th

    Channel 9的官方解释 http://channel9.msdn.com/posts/Inside-the-Azure-Storage-Outage-of-November-18th 曾经在自己 ...

  3. Python sequence (序列)

    序列简介 sequence 是一组有序元素的组合 序列可以是多个元素,也可以一个元素都没有 序列有2种:tuple(定值表).List(表) D:\python\Python_Day>pytho ...

  4. R 中安装xlsx包缺少java环境解决方案

    1.安装Java程序(官网win7 64位系统的Java安装程序及网址http://www.java.com/zh_CN/download/manual.jsp),选择windows 64位脱机安装到 ...

  5. 2天驾驭DIV+CSS (技巧篇)(转)

     这是去年看到的一片文章,感觉在我的学习中,有不少的影响.于是把它分享给想很快了解css的兄弟们.本文是技巧篇. 基础篇[知识一] “DIV+CSS” 的叫法是不准确的[知识二] “DIV+CSS” ...

  6. golang的"..."备忘

    1. 用于数组: 表示长度与元素个数相同. 在golang中数组的长度是类型的一部分,不同长度,不同类型. 2. 用于参数: 用于形参表示可变参数. 用于实参表示直接传递. 具体解释参数见官方文档: ...

  7. CRC校验算法

    typedef unsigned char UCHAR;typedef unsigned char BOOL; /* 计算cnt字节数据的crc,最后一个字节的低7比特必须是0,实际上求的是(cnt× ...

  8. P1179: [Apio2009]Atm

    缩点+spfa最短路,因为最终不可能有环,所以直接spfa. ; type node=record f,t:longint; end; var n,m,s,i,j,ans,cnt,num,u,x,dg ...

  9. 巧用 .NET 中的「合并运算符」获得 URL 中的参数

    获取 URL 中的 GET 参数,无论用什么语言开发网站,几乎是必须会用到的代码.但获取 URL 参数经常需要注意一点就是要先判断是否有这个参数存在,如果存在则取出,如果不存在则用另一个值.这个运算称 ...

  10. 水王ID查找

    一. 题目 1 三人行设计了一个灌水论坛.信息学院的学生都喜欢在上面交流灌水,传说在论坛上有一个“水王”,他不但喜欢发帖,还会回复其他ID发的每个帖子.坊间风闻该“水王”发帖数目超过了帖子数目的一半. ...