Timus 1329. Galactic History。LCA最近公共祖先或dfs递归离线处理!
1329. Galactic History
比赛的时候看到学弟A了这题然后跟榜做,结果在LCA的道路上一去不复返,这个题是很像LCA求最近公共祖先的,不过三个人都没学过LCA,只能拿着资料看着像然后就打上去,结果debug半天,真是吃鸡,边学边做。
题意:n个点,接下来n行每行每个u,v,表示v是u的父节点。v=-1表示u是祖先节点。然后q次查询,每次一个u,v。如果u是v所在的子树的根,输出1,如果v是u所在的子树的根,输出-2,否则输出0。
思路:我们可以先dfs或bfs将这颗树分层,最朴素的思路肯定是从下层往上层查找,基于这个我们想到了LCA的复杂度,如果一层一层找肯定时TLE,亲测有TLE。但我们就想用二分法确定公共祖先,但模板是两个节点同时往上走到同一节点,这题我们需要判断两个节点是否在同一条链中。比赛时间不多,赛后才知道我们dfs递归的时候可以用时间戳来标记节点,如果两个点在同一条链中其时间戳是否一定的关系的,每个点用两个时间戳表示进和出的时间,这样就很好判断了。
int ru[N],chu[N],dep;
vector<int>g[N];
void init()
{
memset(ru,0,sizeof(ru));
memset(chu,0,sizeof(chu));
for(int i=1;i<N;i++) g[i].clear();
}
void dfs(int v,int u)
{
ru[v]=dep++;
for(int i=0;i<g[v].size();i++)
if(g[v][i]!=u) dfs(g[v][i],v);
chu[v]=dep-1;
}
int main()
{
int n,m;
while(~scanf("%d",&n))
{
int root=0,u,v;
init();
for(int i=1;i<=n;i++)
{
scanf("%d%d",&u,&v);
if(v!=-1) g[v].push_back(u);
if(v==-1) root=u;
}
dep=0;
dfs(root,-1);
scanf("%d",&m);
while(m--)
{
scanf("%d%d",&u,&v);
if(ru[u]<=ru[v]&&chu[u]>=chu[v]) puts("1");
else if(ru[u]>=ru[v]&&chu[u]<=chu[v]) puts("2");
else puts("0");
}
}
return 0;
}
思路二:不甘于昨天那种LCA的写法,今天又重新调出来debug了一下终于还是用LCA过了。
和上述说的差不多,LCA求最近公共祖先,从下层往上走,一直逼近,最后到同一层,这个题还没有LCA这么麻烦,我们只需判断从下层往上走到达同一层是否会重合,是的话说明这两个点在同一条链中。先dfs预处理深度和父节点,然后用倍增优化,p[k][v]表示v点往上走2^k层到达的点,预处理出来我们每次就可以在O(logn)内查找了。
vector<int>g[N];
int n,m,p[20][N],d[N],node[N];
void init1()
{
memset(p,0,sizeof(p));
memset(d,0,sizeof(d));
for(int i=0; i<N; i++) g[i].clear();
}
void dfs(int v,int u,int dep)
{
p[0][v]=u;
d[v]=dep;
int len=g[v].size();
for(int j=0; j<len; j++)
if(g[v][j]!=u)
dfs(g[v][j],v,dep+1);
}
void init()
{
for(int k=0; k<17; k++)
for(int i=1; i<=n; i++)
if(p[k][node[i]]<0) p[k+1][node[i]]=-1;
else p[k+1][node[i]]=p[k][p[k][node[i]]];
}
bool lca(int u,int v)
{
for(int k=0; k<17&&d[u]>d[v]&&u!=-1; k++)//倒着来也行,从17到0
{
if((d[u]-d[v])>>k&1)
u=p[k][u];
// else break;
// printf("%d %d %d\n",k,u,p[k][u]);
}
return u==v;
}
int main()
{
while(~scanf("%d",&n))
{
init1();
int u,v,root=0;
for(int i=1; i<=n; i++)
{
scanf("%d%d",&node[i],&v);
if(v!=-1) g[v].push_back(node[i]);
else root=node[i];
}
dfs(root,-1,0);
init();
scanf("%d",&m);
while(m--)
{
scanf("%d%d",&u,&v);
if(d[v]>d[u])
{
if(lca(v,u)) puts("1");
else puts("0");
}
else if(d[u]>d[v])
{
if(lca(u,v)) puts("2");
else puts("0");
}
else puts("0");
}
}
return 0;
}
Timus 1329. Galactic History。LCA最近公共祖先或dfs递归离线处理!的更多相关文章
- lca 最近公共祖先
http://poj.org/problem?id=1330 #include<cstdio> #include<cstring> #include<algorithm& ...
- Tarjan算法应用 (割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)问题)(转载)
Tarjan算法应用 (割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)问题)(转载) 转载自:http://hi.baidu.com/lydrainbowcat/blog/item/2 ...
- LCA(最近公共祖先)模板
Tarjan版本 /* gyt Live up to every day */ #pragma comment(linker,"/STACK:1024000000,1024000000&qu ...
- CodeVs.1036 商务旅行 ( LCA 最近公共祖先 )
CodeVs.1036 商务旅行 ( LCA 最近公共祖先 ) 题意分析 某首都城市的商人要经常到各城镇去做生意,他们按自己的路线去做,目的是为了更好的节约时间. 假设有N个城镇,首都编号为1,商人从 ...
- LCA近期公共祖先
LCA近期公共祖先 该分析转之:http://kmplayer.iteye.com/blog/604518 1,并查集+dfs 对整个树进行深度优先遍历.并在遍历的过程中不断地把一些眼下可能查询到的而 ...
- LCA 近期公共祖先 小结
LCA 近期公共祖先 小结 以poj 1330为例.对LCA的3种经常使用的算法进行介绍,分别为 1. 离线tarjan 2. 基于倍增法的LCA 3. 基于RMQ的LCA 1. 离线tarjan / ...
- LCA最近公共祖先 ST+RMQ在线算法
对于一类题目,是一棵树或者森林,有多次查询,求2点间的距离,可以用LCA来解决. 这一类的问题有2中解决方法.第一种就是tarjan的离线算法,还有一中是基于ST算法的在线算法.复杂度都是O( ...
- Tarjan应用:求割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)【转】【修改】
一.基本概念: 1.割点:若删掉某点后,原连通图分裂为多个子图,则称该点为割点. 2.割点集合:在一个无向连通图中,如果有一个顶点集合,删除这个顶点集合,以及这个集合中所有顶点相关联的边以后,原图变成 ...
- (转)Tarjan应用:求割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)
基本概念: 1.割点:若删掉某点后,原连通图分裂为多个子图,则称该点为割点. 2.割点集合:在一个无向连通图中,如果有一个顶点集合,删除这个顶点集合,以及这个集合中所有顶点相关联的边以后,原图变成多个 ...
随机推荐
- Python3获取大量电影信息:调用API
实验室这段时间要采集电影的信息,给出了一个很大的数据集,数据集包含了4000多个电影名,需要我写一个爬虫来爬取电影名对应的电影信息. 其实在实际运作中,根本就不需要爬虫,只需要一点简单的Python基 ...
- 关于JavaScript的变量和函数提升
第一种理解方式:let和const不能被使用,直到他们被声明 对于var定义的变量,解析器会提升其到作用域顶部. // Outputs: undefined console.log(x); var x ...
- cesium模型加载-加载fbx格式模型
整体思路: fbx格式→dae格式→gltf格式→cesium加载gltf格式模型 具体方法: 1. fbx格式→dae格式 工具:3dsMax, 3dsMax插件:OpenCOLLADA, 下载地址 ...
- UI高端课程
目后佐道IT教育正在打架报名中,欢迎高中生.大学生前来学习编程技术和UI设计,招生面向全国. 石破天惊 前100个报名者免费提供高级公寓居住(里面有空调,暖气,热水器,洗衣机). 赠送神秘课程价值29 ...
- 10.1 plan
1951 [Sdoi2010]古代猪文 Sdoi2010 Contest2 807 1928 1566 [NOI2009]管道取珠 806 1429 2756 [S ...
- xcdatamodel的实质
修改后缀名为zip或者其它,可以查看到xcdatamodel是一个描述文件 <?xml version="1.0" encoding="UTF-8" st ...
- BCB:Windows消息处理
Windows消息处理 BCB 本文研究了BCB中的消息处理机制,在此基础上提出了处理Windows消息和自定义消息响应的方法和建立动态和静态消息映射的技巧. C++ Builder作为一种RAD方式 ...
- CPP-基础:C/C++数组名与指针的区别
2005-08-23 08:36 来源:天极网 作者:宋宝华 责任编辑:方舟·yesky 引言 指针是C/C++语言的特色,而数组名与指针有太多的相似,甚至很多时候,数组名可以作为指针使用.于是乎,很 ...
- springboot超详细笔记
一.Spring Boot 入门 1.Spring Boot 简介 简化Spring应用开发的一个框架: 整个Spring技术栈的一个大整合: J2EE开发的一站式解决方案: 2.微服务 2014,m ...
- Linux下Jenkins与GitHub自动构建Node项目(Vue)
根据上篇文章<Linux下Jenkins与GitHub自动构建NetCore与部署>,我们知道了Jenkins的强大功能,自动构建,部署了一个NetCore的Web,让开发人员专注于开发, ...