题目大意:给你一棵树,有两个操作1.修改一条边的值,2.询问从x到y路径上边的最大值

思路:如果树退化成一条链的话线段树就很明显了,然后这题就是套了个树连剖分,调了很久终于调出来第一个模板了

 #include<iostream>
#include<cstdio>
#include<cstring>
#define maxn 100009
using namespace std;
int head[maxn],next[maxn*],point[maxn],son[maxn],size_k[maxn],id[maxn],value[maxn],now=,father[maxn],top[maxn];
int x[maxn],y[maxn],v[maxn],tree[maxn*],pos=,deep[maxn],n;
int read()
{
int x=,f=;char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();}
while(ch>=''&&ch<=''){x=x*+ch-'';ch=getchar();}
return x*f;
}
void add(int x,int y,int v)
{
next[++now]=head[x];head[x]=now;
point[now]=y;value[now]=v;
}
int dfs(int k,int fa)
{
deep[k]=deep[fa]+;
father[k]=fa;
size_k[k]=;
int max_x=-;
son[k]=-;
for(int i=head[k];i;i=next[i])
{
if(point[i]==fa)continue;
dfs(point[i],k);
size_k[k]+=size_k[point[i]];
if(size_k[point[i]]>max_x)
{
max_x=size_k[point[i]];
son[k]=point[i];
}
}
}
void dfs2(int k,int fa,int pa)
{
id[k]=++pos;
top[k]=pa;
if(son[k]!=-)dfs2(son[k],k,pa);
for(int i=head[k];i;i=next[i])
{
if(point[i]!=fa && point[i]!=son[k])dfs2(point[i],k,point[i]);
}
}
void update(int node,int l,int r,int pos,int x)
{
if(l+==r){tree[node]=x;return;}
int mid=(l+r)>>;
if(pos<mid)update(node*,l,mid,pos,x);
else update(node*+,mid,r,pos,x);
tree[node]=max(tree[node*],tree[node*+]);
}
int query(int node,int l,int r,int ql,int qr)
{
if(ql<=l && r<=qr)return tree[node];
int mid=(l+r)>>;
int ans=-0x3f3f3f3f;
if(ql<mid)ans=max(query(node<<,l,mid,ql,qr),ans);
if(mid<qr)ans=max(query(node<<|,mid,r,ql,qr),ans);
return ans;
}
int find(int x,int y)
{
int fa=top[x],fb=top[y],temp=;
while(fa!=fb)
{
if(deep[fa]<deep[fb])
{
swap(fa,fb);
swap(x,y);
}
if(id[fa]>id[x]+)return -;
temp=max(temp,query(,,pos+,id[fa],id[x]+));
x=father[fa];fa=top[x];
}
if(x==y)return temp;
if(deep[x]<deep[y])swap(x,y);
if(id[y]+>id[x]+)return -;
return max(temp,query(,,pos+,id[son[y]],id[x]+));
}
int main()
{
int t;
t=read();
while(t--)
{
now=;
memset(head,,sizeof(head));
memset(tree,,sizeof(tree));
n=read();
for(int i=;i<n;i++)
{
x[i]=read();y[i]=read();v[i]=read();
add(x[i],y[i],v[i]);
add(y[i],x[i],v[i]);
}
deep[]=;
dfs(,);
dfs2(,,);
for(int i=;i<=n-;i++)
{
int u=deep[x[i]]>deep[y[i]]?x[i]:y[i];
update(,,pos+,id[u],v[i]);
}
int xx,yy;
char ch[];
while(true)
{
scanf("%s",ch);
if(ch[]=='D')break;
xx=read();yy=read();
if(ch[]=='Q')
{
printf("%d\n",find(xx,yy));
}
else if(ch[]=='C')
{
int u=deep[x[xx]]>deep[y[xx]]?x[xx]:y[xx];
update(,,pos+,id[u],yy);
}
else if(ch[]=='D')
{
break;
}
}
}
return ;
}

SPOJ 375 Query on a tree【树链剖分】的更多相关文章

  1. spoj 375 Query on a tree (树链剖分)

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

  2. SPOJ 375 Query on a tree 树链剖分模板

    第一次写树剖~ #include<iostream> #include<cstring> #include<cstdio> #define L(u) u<&l ...

  3. SPOJ QTREE Query on a tree 树链剖分+线段树

    题目链接:http://www.spoj.com/problems/QTREE/en/ QTREE - Query on a tree #tree You are given a tree (an a ...

  4. spoj QTREE - Query on a tree(树链剖分+线段树单点更新,区间查询)

    传送门:Problem QTREE https://www.cnblogs.com/violet-acmer/p/9711441.html 题解: 树链剖分的模板题,看代码比看文字解析理解来的快~~~ ...

  5. SPOJ QTREE Query on a tree ——树链剖分 线段树

    [题目分析] 垃圾vjudge又挂了. 树链剖分裸题. 垃圾spoj,交了好几次,基本没改动却过了. [代码](自带常数,是别人的2倍左右) #include <cstdio> #incl ...

  6. SPOJ QTREE Query on a tree --树链剖分

    题意:给一棵树,每次更新某条边或者查询u->v路径上的边权最大值. 解法:做过上一题,这题就没太大问题了,以终点的标号作为边的标号,因为dfs只能给点分配位置,而一棵树每条树边的终点只有一个. ...

  7. spoj 375 QTREE - Query on a tree 树链剖分

    题目链接 给一棵树, 每条边有权值, 两种操作, 一种是将一条边的权值改变, 一种是询问u到v路径上最大的边的权值. 树链剖分模板. #include <iostream> #includ ...

  8. SPOJ Query on a tree 树链剖分 水题

    You are given a tree (an acyclic undirected connected graph) with N nodes, and edges numbered 1, 2, ...

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

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

  10. Bzoj 2588 Spoj 10628. Count on a tree(树链剖分LCA+主席树)

    2588: Spoj 10628. Count on a tree Time Limit: 12 Sec Memory Limit: 128 MB Description 给定一棵N个节点的树,每个点 ...

随机推荐

  1. git的基本使用命令操作

    Linux操作命令行:    mkdir - 创建文件夹,    cd - 切换文件路径    pwd - 显示文件路径    ls -ah - 可以查看隐藏的文件夹名(.git)    cat 文件 ...

  2. JavaScript 跨域请求

    1.最简单通用的做法就是 反向代理         通过nginx搭建一个反向代理服务器,通过将跨域的请求配置成转发:此方法适用于动静分离时,很多跨域请求的情况下: server { listen 8 ...

  3. Cordova应用的JavaScript代码和自定义插件代码的调试

    我之前写过三篇Cordova相关的技术文章.当我们使用Cordova将自己开发的前端应用打包安装到手机上后,可能会遇到需要调试Cordova应用的时候. 本文就介绍Cordova应用的调试步骤. 如果 ...

  4. 9.18 New Start

    好久没上cnblogs,今天提示我说园龄已经2年1个月了.今天就用一个日记的形式开始第一篇博客吧.我以后比较精髓的文章就放在cnblogs,csdn博客也继续会更新,不过也会慢慢提高文章质量. 今天是 ...

  5. (转)SpringMVC学习(四)——Spring、MyBatis和SpringMVC的整合

    http://blog.csdn.net/yerenyuan_pku/article/details/72231763 之前我整合了Spring和MyBatis这两个框架,不会的可以看我的文章MyBa ...

  6. URAL 1776 Anniversary Firework (概率,区间DP)

    坑,一开始以为,分成两半的时候去最大那个就行了, 实际上这样是不对的,因为有可能出现小的一半的时间比大的要长, 因为还和等待次数有关,且转移的时候需要用到次数更小的状态, 所以状态定义为二维,dp[i ...

  7. HDOJ1195 双向BFS //单向也可以过 没想清

    #include<cstdio> #include<map> #include<vector> #include<stack> #include< ...

  8. Android(java)学习笔记144:网络图片浏览器的实现(ANR)

    1.我们在Android下,实现使用http协议进行网络通信,请求网络数据.这里是获取网络上的图片信息,让它可以显示在手机上: 但是我们这个手机连接网络是很费时间,如果我们在主线程(UI线程)中写这个 ...

  9. SVN中的check out与export的区别

    http://blog.csdn.net/zndxlxm/article/details/7763116 check out跟check in对应,export跟import对应. check out ...

  10. WPF知识点全攻略05- XAML内容控件

    此处简单列举出布局控件外,其他常用的控件: Window:WPF窗口 UserControl:用户控件 Page:页 Frame:用来浏览Page页 Border:嵌套控件,提供边框和背景. Butt ...