lca的好题!网上用st表和离线解的比较多,用树上倍增也是可以做的

不知道错在哪里,等刷完了这个专题再回来看

题解链接https://blog.csdn.net/Sd_Invol/article/details/9572423

/*
给一颗点权树,求出一个点对(x,y)之间的max{A,B,C}
A:x到lca路径上的最大差值
B:lca到y路径上的最大差值
C:x到y路径上的最大差值
需要维护的值,x结点到的祖先,x结点到祖先路径上的最大值,最小值,x结点到路径上的最大收益,最小收益(可以是负数)
对于每个询问x->y:A:x到祖先的最大收益
B:y到祖先的最小收益的负值
C:x到lca的最大值减去lca到y的最小值
*/
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<algorithm>
using namespace std;
#define maxn 50005
struct Edge{
int x,next;
}e[maxn<<];
int head[maxn],tot,n,q;
inline void addedge(int u,int v){
e[tot].x=v;
e[tot].next=head[u];
head[u]=tot++;
}
int d[maxn],f[maxn][],a[maxn];//深度,祖先,点权
int fm[maxn][],fn[maxn][],sm[maxn][],sn[maxn][];//最大值,最小值,最大收益,最小收益
void init(){
tot=;
memset(head,-,sizeof head);
}
void dfs(int x,int fa,int dep){
d[x]=dep,f[x][]=fa;
for(int i=head[x];i!=-;i=e[i].next)
if(fa!=e[i].x) dfs(e[i].x,x,dep+);
}
int query(int x,int y){
int i,xx=,yy=,X=a[x],Y=a[y];//x侧最大收益,y侧最小收益,x侧最小值,y侧最大值
i=;
while(d[x]!=d[y]){//拉倒同一高度
if(abs(d[x]-d[y]) >= <<i)
if(d[y]<d[x])
xx=max(max(xx,sm[x][i]),fm[x][i]-X),X=min(X,fn[x][i]),x=f[x][i];
else
yy=min(min(yy,sn[y][i]),fn[y][i]-Y),Y=max(Y,fm[y][i]),y=f[y][i];
--i;
}
if(x==y) return max(max(xx,-yy),Y-X);
i=;
while(i>=){
if(f[x][i] && f[y][i] && f[x][i]!=f[y][i]){
xx=max(max(xx,sm[x][i]),fm[x][i]-X),X=min(X,fn[x][i]),x=f[x][i];
yy=min(min(yy,sn[y][i]),fn[y][i]-Y),Y=max(Y,fm[y][i]),y=f[y][i];
}
--i;
}
i=;//这里还要跳一次
xx=max(max(xx,sm[x][i]),fm[x][i]-X),X=min(X,fn[x][i]),x=f[x][i];
yy=min(min(yy,sn[y][i]),fn[y][i]-Y),Y=max(Y,fm[y][i]),y=f[y][i];
return max(max(xx,-yy),Y-X);
}
void work(){
int x,y;
init();
scanf("%d",&n);
for(int i=;i<=n;i++) scanf("%d",&a[i]);
for(int i=;i<n;i++){
scanf("%d%d",&x,&y);
addedge(x,y);addedge(y,x);
}
dfs(,,);
fm[][]=fn[][]=a[];
sm[][]=-<<;sn[][]=<<;
for(int i=;i<=n;i++){//先打初始状态
fm[i][]=max(a[i],a[f[i][]]);//和父亲比较
fn[i][]=min(a[i],a[f[i][]]);
sm[i][]=max(,a[f[i][]]-a[i]);//要么是0,要么赚了钱
sn[i][]=min(,a[f[i][]]-a[i]);//要么是0,要么是亏了钱
}
for(int j=;(<<j)<n;j++)//再打剩下状态
for(int i=;i<=n;i++){
int tmp=f[i][j-];//中间态
f[i][j]=f[tmp][j-];
fm[i][j]=max(fm[i][j-],fm[tmp][j-]);
fn[i][j]=min(fm[i][j-],fm[tmp][j-]);
sm[i][j]=max(max(sm[i][j-],sm[tmp][j-]),fm[tmp][j-]-fn[i][j-]);//最大收益要么是两段间的最大收益,要么是祖先段的最大收益减去子孙段的最小收益
sn[i][j]=min(min(sn[i][j-],sn[tmp][j-]),fn[tmp][j-]-fm[i][j-]);//最小收益相反
}
scanf("%d",&q);
while(q--){
scanf("%d%d",&x,&y);
printf("%d\n",query(x,y));
}
}
int main(){
work();
return ;
}

poj3728 倍增法lca 好题!的更多相关文章

  1. Codeforces 519E A and B and Lecture Rooms [倍增法LCA]

    题意: 给你一棵有n个节点的树,给你m次询问,查询给两个点,问树上有多少个点到这两个点的距离是相等的.树上所有边的边权是1. 思路: 很容易想到通过记录dep和找到lca来找到两个点之间的距离,然后分 ...

  2. 倍增法-lca codevs 1036 商务旅行

    codevs 1036 商务旅行  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond 题目描述 Description 某首都城市的商人要经常到各城镇去做生意 ...

  3. 倍增法lca

    ][N],siz[N];//rt数组需要在dfs之前置-1. void dfs(int pos,int deep){ dep[pos]=deep; siz[pos]=; for(edge *it=ad ...

  4. poj1470 LCA倍增法

    倍增法模板题 #include<iostream> #include<cstring> #include<cstdio> #include<queue> ...

  5. 最近公共祖先算法LCA笔记(树上倍增法)

    Update: 2019.7.15更新 万分感谢[宁信]大佬,认认真真地审核了本文章,指出了超过五处错误捂脸,太尴尬了. 万分感谢[宁信]大佬,认认真真地审核了本文章,指出了超过五处错误捂脸,太尴尬了 ...

  6. 浅谈倍增法求解LCA

    Luogu P3379 最近公共祖先 原题展现 题目描述 如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先. 输入格式 第一行包含三个正整数 \(N,M,S\),分别表示树的结点个数.询问 ...

  7. lca入门———树上倍增法(博文内含例题)

    倍增求LCA: father[i][j]表示节点i往上跳2^j次后的节点 可以转移为 father[i][j]=father[father[i][j-1]][j-1] 整体思路: 先比较两个点的深度, ...

  8. 倍增法求LCA(最近公共最先)

    对于有根树T的两个结点u.v,最近公共祖先x=LCA(u,v)表示一个结点x,满足x是u.v的祖先且x的深度尽可能大. 如图,根据定义可以看出14和15的最近公共祖先是10,   15和16的最近公共 ...

  9. hdu2586 lca倍增法

    倍增法加了边的权值,bfs的时候顺便把每个点深度求出来即可 #include<iostream> #include<cstring> #include<cstdio> ...

随机推荐

  1. Linux命令之ll

    ll命令 用处:以长格形式列出当前目录下的所有文件,每个文件的长度和创建时间不同. 用法:输入 ll 示例: 前面的一大串字母的意思,第一个要么是d要么是-,d的意思就是目录,-的意思就是文件.其后的 ...

  2. 学习windows编程 day3 之滚动条完善

    1.不再使用setscrollrange,setscrollpos,getscrollrange,getscrollpos这些函数,这只是有助于理解其中运行原理 2.改用setscrollinfo,g ...

  3. js jquery获取当前元素的兄弟级 上一个 下一个元素

    原博地址:http://www.jb51.net/article/71782.htm var chils= s.childNodes;  //得到s的全部子节点 var par=s.parentNod ...

  4. java内存模型及内存与cpu之间的关系

    主内存和cpu之间的关系,因为cpu是在是处理速度太快了.所以一般cpu都有一个cpu缓存,上图的意思是主内存--->cpu缓存--->cpu寄存器--->cpu执行处理,写的时候反 ...

  5. javascript 模拟京东关闭广告栏

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  6. 【收藏】轻松导出全民K歌里任何人录制的短视频(MV)、歌曲的方法

    有一次想把她在全民K歌里唱过的所有歌下载到电脑上,然后合成一个视频.但不知道怎么导出全民K歌里的歌曲,经过各种百度Google终于找到了一个用起来很简单的工具.不仅可以下载保存任何人录制的歌曲,还可以 ...

  7. JAVA-大白话探索JVM-运行时内存(三)

    前面章节 JAVA-大白话探索JVM-类加载器(一) JAVA-大白话探索JVM-类加载过程(二) JVM运行时内存 通过之前的章节,我们知道.class类如何加载到内存中,如图红框 开始讲讲内存空间 ...

  8. CentOS6.8下MySQL数据库忘记root密码解决方法

    一.更改my.cnf配置文件 编辑/etc/my.cnf文件,在[mysqld]下添加skip-grant-tables,保存退出.如图: 重启mysql服务:services mysqld rest ...

  9. Java编程思想 学习笔记3

    三.操作符 1.优先级 当一个表达式中存在多个操作符时,操作符的优先级就决定了各部分的计算顺序.程序员常常忘记其他优先级规则,所以应该用括号明确规定计算顺序. 当编译器观察到一个String后面紧跟着 ...

  10. WebSocket实战之——JavaScript例子

    一.详细代码案例 详细解读一个简单html5 WebSocket的Js实例教程,附带完整的javascript websocket实例源码,以及实例代码效果演示页面,并对本实例的核心代码进行了深入解读 ...