题意简单明了(这就是个模板)。

就是让我们找2个节点的公共祖先而已,但我们要讲的做法不是生硬的爆搜,而且直接搜好像过不去……

这次就讲我往后拖了n多天才开始学了倍增LCA。

嗯,这个题,如果2个节点的深度是不一样的,我们要把他们的深度变成一样的,变成一样的以后就开始倍增搜索。

上面的这句话为我们点明了这个题的解题方法,我们要求出每个节点的深度,每个节点2的几次方个祖先是谁,还有这棵树是什么样子的(一开始我把第一个输入的当成爸爸,wa的老惨了)。

首先第一步:创建这颗树

题目中给出了树的根节点,所以说,和根节点连接的就是第二层的节点,和第二次节点连接的就是第三层的节点……和第n-1层连接的就是第n层的节点。

虽然这个的实现很简单,但如果有同学不会,可以戳这里

树建完了,开始第二步:查找一个节点第2的i次方个祖先是谁?

有个神奇的东西叫做RMQ,和这个差不多,懂了可以去秒一下。

我们可以通过一个生活实例来思考,比如你爸爸就是你第2^0个祖先,你第2^1的祖先就是你爸爸的爸爸,你的第2^2个祖先就是你爷爷的爷爷……

可以发现,我们要求的东西可以通过已知的东西获得,这就避免了一些奇怪的模拟。

现在树建完了,每个子节点的祖先也知道了,接下来就是倍增模拟了:

就像上面说的一样,先把2个位置统一公共祖先,然后寻找。

思路都讲完了,直接放代码了:

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<cstring>
using namespace std;
long long n,m,s,a,b;
long long cf[100],wz,shu=1;
long long lca[500005][30],sd[500005];
long long ans,head[500005];
struct hehe
{
long long nxt,wei;
}sz[1000005];
int add(int tou,int wei)//链式前向星真香
{
ans++;
sz[ans].wei=wei;
sz[ans].nxt=head[tou];
head[tou]=ans;
}
int dfs(int qd,int bb)//搜深度的地方
{
lca[qd][0]=bb;
for(int i=1;(1<<i)<=sd[qd];i++)//没必要就不搜了,很明白的道理嘛
{
lca[qd][i]=lca[lca[qd][i-1]][i-1];//想想祖孙三代的例子或许就明白这句话了。
}
for(int i=head[qd];i!=0;i=sz[i].nxt)//链式前向星的查找
{
if(sz[i].wei!=bb)//因为不知道谁是谁的爸爸,所以要特判一下。
{
sd[sz[i].wei]=sd[qd]+1;
dfs(sz[i].wei,qd);
}
}
}
long long start(long long x,long long y)
{
if(sd[x]<sd[y])//永远让x的深度
{
swap(x,y);
}
int i=0;
while((1<<i)<=sd[x])
{
i++;
}
i--;
if(sd[x]!=sd[y])//不等于就大步的往前走
{
for(int j=i;j>=0;j--)
{
if(sd[lca[x][j]]>=sd[y])//小心不要走过。
{
x=lca[x][j];
}
}
}
if(x==y)//x是y的子节点呢,没事了,直接输出。
{
return x;
}
for(int j=i;j>=0;j--)
{
if(lca[x][j]!=lca[y][j])//这个依旧不要走过
{
x=lca[x][j];
y=lca[y][j];
}
}
return lca[x][0];//因为倍增的奇妙,最后一定会停留在答案的下一层,上升一层就可以了。
}
int main()
{
scanf("%lld%lld%lld",&n,&m,&s);//正常的读入
sd[s]=1;
lca[s][0]=0;
for(int i=0;i<n-1;i++)
{
scanf("%lld%lld",&a,&b);
add(a,b);
add(b,a);
}
dfs(s,0);
for(int i=0;i<m;i++)
{
scanf("%lld%lld",&a,&b);
printf("%lld\n",start(a,b));//正常的运算和输出
}
return 0;
}

这个题就到这里吧,讲完了。

P3379 最近公共祖先(LCA) 洛谷的更多相关文章

  1. Luogu 2245 星际导航(最小生成树,最近公共祖先LCA,并查集)

    Luogu 2245 星际导航(最小生成树,最近公共祖先LCA,并查集) Description sideman做好了回到Gliese 星球的硬件准备,但是sideman的导航系统还没有完全设计好.为 ...

  2. POJ 1470 Closest Common Ancestors(最近公共祖先 LCA)

    POJ 1470 Closest Common Ancestors(最近公共祖先 LCA) Description Write a program that takes as input a root ...

  3. POJ 1330 Nearest Common Ancestors / UVALive 2525 Nearest Common Ancestors (最近公共祖先LCA)

    POJ 1330 Nearest Common Ancestors / UVALive 2525 Nearest Common Ancestors (最近公共祖先LCA) Description A ...

  4. [模板] 最近公共祖先/lca

    简介 最近公共祖先 \(lca(a,b)\) 指的是a到根的路径和b到n的路径的深度最大的公共点. 定理. 以 \(r\) 为根的树上的路径 \((a,b) = (r,a) + (r,b) - 2 * ...

  5. 【lhyaaa】最近公共祖先LCA——倍增!!!

    高级的算法——倍增!!! 根据LCA的定义,我们可以知道假如有两个节点x和y,则LCA(x,y)是 x 到根的路 径与 y 到根的路径的交汇点,同时也是 x 和 y 之间所有路径中深度最小的节 点,所 ...

  6. P3379最近公共祖先(LCA)

    题目描述 如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先. 输入输出格式 输入格式: 第一行包含三个正整数N.M.S,分别表示树的结点个数.询问的个数和树根结点的序号. 接下来N-1行每 ...

  7. 最近公共祖先 lca (施工ing)

    声明 咳咳,进入重难点的图论算法之一(敲黑板): 题目: 洛谷 P3379 先放标程,施工ing,以后补坑!!!(实在太难,一个模板这么长 [ 不过好像还是没有 AC自动机 长哎 ],注释都打半天,思 ...

  8. 图论2 最近公共祖先LCA

    模板 吸取洛谷P3379的教训,我决定换板子(其实本质都是倍增是一样的),把vector换成了边表 输入格式: 第一行包含三个正整数N.M.S,分别表示树的结点个数.询问的个数和树根结点的序号. 接下 ...

  9. 求最近公共祖先(LCA)的各种算法

    水一发题解. 我只是想存一下树剖LCA的代码...... 以洛谷上的这个模板为例:P3379 [模板]最近公共祖先(LCA) 1.朴素LCA 就像做模拟题一样,先dfs找到基本信息:每个节点的父亲.深 ...

随机推荐

  1. Python 网络爬虫实战:爬取 B站《全职高手》20万条评论数据

    本周我们的目标是:B站(哔哩哔哩弹幕网 https://www.bilibili.com )视频评论数据. 我们都知道,B站有很多号称“镇站之宝”的视频,拥有着数量极其恐怖的评论和弹幕.所以这次我们的 ...

  2. Java 多线程基础(十)interrupt()和线程终止方式

    Java 多线程基础(十)interrupt()和线程终止方式 一.interrupt() 介绍 interrupt() 定义在 Thread 类中,作用是中断本线程. 本线程中断自己是被允许的:其它 ...

  3. git merge整理

    ========================================================== git bash merge 一.开发分支(dev)上的代码达到上线的标准后,要合 ...

  4. Java 将PDF/XPS转为Word/html /SVG/PS/PCL/PNG、PDF和XPS互转(基于Spire.Cloud.SDK for Java)

    Spire.Cloud.SDK for Java提供了接口PdfConvertApi通过convert()方法将PDF文档以及XPS文档转为指定文档格式,如转PDF为Word(支持Docx.Doc). ...

  5. mybatis缓存之一级缓存(一)

    对于mybatis框架.仿佛工作中一直是在copy着使用.对于mybatis缓存.并没有一个准确的认知.趁着假期.学习下mybatis的缓存.这篇主要学习mybatis的一级缓存. 为什么使用缓存 其 ...

  6. Java BigDecimal和double BigDecimal类

    BigDecimal类 对于不需要任何准确计算精度的数字可以直接使用float或double,但是如果需要精确计算的结果,则必须使用BigDecimal类,而且使用BigDecimal类也可以进行大数 ...

  7. 数据库周刊30丨数据安全法草案将亮相;2020数据库产业报告;云南电网上线达梦;达梦7误删Redo Log;Oracle存储过程性能瓶颈;易鲸捷实践案例……

    摘要:墨天轮数据库周刊第30期发布啦,每周1次推送本周数据库相关热门资讯.精选文章.干货文档. 热门资讯 1.数据安全法草案即将亮相:将确立数据分级分类管理.应急处置制度[摘要]数据安全法草案即将在本 ...

  8. jupyter lab最强代码补全插件

    1 简介 提起kite相信不少朋友都有印象,它是一个功能非常强大的代码补全工具,目前可用于Python与javascript,为许多知名的编辑器譬如Vs Code.Pycharm提供对应的插件. 图1 ...

  9. 八.django日志配置

    Django 日志 Django 使用Python 内建的logging 模块打印日志,Python 的logging 配置由四个部分组成: 记录器 —— Logger 处理程序 —— Handler ...

  10. 图灵学院JAVA互联网架构师专题学习笔记

    图灵学院JAVA互联网架构师专题学习笔记 下载链接:链接: https://pan.baidu.com/s/1xbxDzmnQudnYtMt5Ce1ONQ 密码: fbdj如果失效联系v:itit11 ...