QTREE - Query on a tree
QTREE - Query on a tree
题目链接:http://www.spoj.com/problems/QTREE/
参考博客:http://blog.sina.com.cn/s/blog_7a1746820100wp67.html
树链剖分入门题
代码如下(附注解):
#include <cstdio>
#include <cstring>
#include <iostream>
#define lson (x<<1)
#define rson ((x<<1)|1)
#define mid ((l+r)>>1)
#define N 10010
using namespace std;
struct Edge{
int to,next;
}e[N<<];
int T,n,tot,oo,root;
],tree[N<<];//头结点,inpotData,线段树
int fa[N],siz[N],dep[N],son[N];//父亲,包含该节点的子树的结点数,深度,重链上的儿子
int top[N],p[N];//当前节点所在重链的顶端结点,当前节点与其父亲的连边在线段树中的位置
];
void addEdge(int u,int v){//用头接法链表存当前节点的各个儿子
e[++oo].to=v;
e[oo].next=head[u];
head[u]=oo;
}
void fristDFS(int v){//第一次dfs求siz,son,dep,fa
siz[v]=,son[v]=;
;i=e[i].next){
int u=e[i].to;
if(u!=fa[v]){
fa[u]=v;
dep[u]=dep[v]+;
fristDFS(u);
if(siz[u]>siz[son[v]])son[v]=u;
siz[v]+=siz[u];
}
}
}
void secondDFS(int v,int tp){//第二次dfs求top,p
p[v]=++tot,top[v]=tp;
)secondDFS(son[v],top[v]);
;i=e[i].next){
int u=e[i].to;
if(u!=fa[v]&&u!=son[v])secondDFS(u,u);
}
}
void push_up(int x){
tree[x]=max(tree[lson],tree[rson]);
}
void updata(int x,int l,int r,int p,int v){//建线段树
if(l==r){
tree[x]=v;
return;
}
if(p<=mid)updata(lson,l,mid,p,v);
,r,p,v);
push_up(x);
}
void init(){
memset(siz,,sizeof(siz));
memset(head,,sizeof(head));
scanf("%d",&n);
root=(n+)/;//任意取一个root
oo=tot=fa[root]=dep[root]=;
;i<n;++i){
;j<;++j)scanf("%d",&a[i][j]);
addEdge(a[i][],a[i][]);
addEdge(a[i][],a[i][]);
}
fristDFS(root);
secondDFS(root,root);
;i<n;++i){
]]>dep[a[i][]])swap(a[i][],a[i][]);
//将远离root的结点放在d[i][1]
updata(,,tot,p[a[i][]],a[i][]);
}
}
int query(int x,int l,int r,int a,int b){
if(a<=l&&r<=b)return tree[x];
;
if(a<=mid)tmp=max(tmp,query(lson,l,mid,a,b));
,r,a,b));
return tmp;
}
int find(int x,int y){
;
while(f1!=f2){//若x,y不在同一条重链上
if(dep[f1]<dep[f2])swap(f1,f2),swap(x,y);
//取深度大的点,f1为该重边的顶端结点,va为该结点
//在线段树中[f1,va]一定为一个连续的区间
tmp=max(tmp,query(,,tot,p[f1],p[x]));
x=fa[f1];f1=top[x];
}
if(x==y)return tmp;
if(dep[y]<dep[x])swap(x,y);
,,tot,p[son[x]],p[y]));//不包含x与其父亲构成的边
}
void solve(){
int x,y;
]!='D';scanf("%s",s)){
scanf("%d%d",&x,&y);
]=='Q')printf("%d\n",find(x,y));
]==,,tot,p[a[x][]],y);
}
}
int main(void){
for(scanf("%d",&T);T;T--){
init();
solve();
}
}
QTREE - Query on a tree的更多相关文章
- 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 ...
- SP375 QTREE - Query on a tree (树剖)
题目 SP375 QTREE - Query on a tree 解析 也就是个蓝题,因为比较长 树剖裸题(基本上),单点修改,链上查询. 顺便来说一下链上操作时如何将边上的操作转化为点上的操作: 可 ...
- SPOJ VJudge QTREE - Query on a tree
Query on a tree Time Limit: 851MS Memory Limit: 1572864KB 64bit IO Format: %lld & %llu Submi ...
- spoj QTREE - Query on a tree(树链剖分+线段树单点更新,区间查询)
传送门:Problem QTREE https://www.cnblogs.com/violet-acmer/p/9711441.html 题解: 树链剖分的模板题,看代码比看文字解析理解来的快~~~ ...
- SPOJ QTREE Query on a tree --树链剖分
题意:给一棵树,每次更新某条边或者查询u->v路径上的边权最大值. 解法:做过上一题,这题就没太大问题了,以终点的标号作为边的标号,因为dfs只能给点分配位置,而一棵树每条树边的终点只有一个. ...
- SPOJ375 QTREE - Query on a tree
本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/ ...
- SP375 QTREE - Query on a tree
题意大意 给定\(n\)个点的树,边按输入顺序编号为\(1,2,...n-1\),要求作以下操作: CHANGE \(i\) \(t_i\) 将第\(i\)条边权值改为\(t_i\),QUERY \( ...
- SPOJ QTREE Query on a tree VI
You are given a tree (an acyclic undirected connected graph) with n nodes. The tree nodes are number ...
- SPOJ QTREE Query on a tree V
You are given a tree (an acyclic undirected connected graph) with N nodes. The tree nodes are number ...
随机推荐
- C#串口通信程序详解
C#串口通信程序详解 摘要:创建C#串口通信程序需要注意什么呢?创建C#串口通信程序的步骤是什么?那么本文就向你详细介绍创建C#串口通信程序集体的内容. 在.NET平台下创建C#串口通信程序,.NET ...
- try { var mergeFilePath = string.Format("{0}mergepdf.pdf", tempDownDir); PDFPrintHelper.MergePDFFile(pdfList, mergeFi
winform 按顺序连续打印多个PDF文件 关于PDF打印的问题,前面有篇文章(点这里查看)也叙述过,今天来谈谈另外一种方法 其实方法很简单,因为需要把多个PDF文档按顺序连续打印,为此我们为什 ...
- [转]About the security content of iOS 8
Source:http://support.apple.com/kb/HT6441 For the protection of our customers, Apple does not disclo ...
- C#制作高仿360安全卫士窗体3
C#制作高仿360安全卫士窗体(三) 距上篇C#制作高仿360安全卫士窗体(二)也将近一个多月了,这个月事情还是像往常一样的多.不多我也乐在其中,毕竟我做的是我喜欢做的东西.今天特地抽空把怎么制作 ...
- 不高级不能发帖的WPS论坛
今天又发现了一个难用到令人发指的社区:WPS论坛.它的产品经理一定没用过这个论坛或者它根本没有产品经理. 发帖提示悬赏分不能为0,但整个界面就没有悬赏分有关的东西,于是尝试点击下面的快速回复,结果导致 ...
- MySQL 常用命令大全
Mysql常用命令行大全 第一招.mysql服务的启动和停止 net stop mysql net start mysql 第二招.登陆mysql 语法如下: mysql -u用户名 -p用户密码 键 ...
- javascript ajax 脚本跨域调用全解析
javascript ajax 脚本跨域调用全解析 今天终于有点时间研究了一下javsscript ajax 脚本跨域调用的问题,先在网上随便搜了一下找到一些解决的办法,但是都比较复杂.由是转到jqu ...
- 【原】tinker dex文件格式的dump工具tinker-dex-dump
序言 Tinker是微信推出的热更新开源项目,同其它热更新方案相比具有补丁包小,支持类,so,资源文件的替换等优点.其中在类替换的方案里自主研发了DexDiff算法,使得补丁包变的更小.DexDiff ...
- C语言之scarf函数
一 基本用法 scanf函数:接收用户的输入 语法: scanf("格式化控制符",地址列表); 例: int num; scanf("%d",&num ...
- 怎样求FIRST集、FOLLOW集和SELECT集
一,要知道什么是终结符和非终结符. 终结符:通俗的说就是不能单独出现在推导式左边的符号,也就是说终结符不能再进行推导. 非终结符:不是终结符的都是非终结符.(非男即女,呵呵) 如:A-->B,则 ...