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.割点集合:在一个无向连通图中,如果有一个顶点集合,删除这个顶点集合,以及这个集合中所有顶点相关联的边以后,原图变成多个 ...
随机推荐
- 编写C#程序,自动将bing首页图片设为壁纸
任务目标: 1,获取图片 2,设为壁纸 3,自动化 环境需求: .NET Framework 4.0+, Visual Studio 2017 ==================== 1,获取图片 ...
- Mongodb之failed to create service entry worker thread
Mongodb "failed to create service entry worker thread" 错误. 系统:CentOS release 6.8 mongod.lo ...
- centos7中文显示为小方块~~啊啊啊 求大佬们解答
这个问题困扰我很久了,刚好前几天注册了博客园,就想问问大佬们是怎么解决中文显示小方块的? 我试了很多办法,包括但不限于修改i18n配置文件,locale.conf,添加中文字体库等等等... 但都没有 ...
- JQuery EasyUI学习记录(三)
1.jQuery EasyUI messager使用方式 1.1 alert方法 $(function(){ //1.alert方法---提示框 $.messager.alert("标题&q ...
- 洛谷 P1147 连续自然数和
洛谷 P1147 连续自然数和 看到dalao们的各种高深方法,本蒟蒻一个都没看懂... 于是,我来发一篇蒟蒻友好型的简单题解 #include<bits/stdc++.h> using ...
- MySQL 使用GTID进行复制
MySQL 使用GTID进行复制 1. GTID的格式和存储 1.1 GTID 集 1.2 mysql.gtid_executed 表 1.3 mysql.gtid_executed 表压缩 2. G ...
- 笔记--Day1--python基础1
一.目录 1.Python介绍 python的创始人为吉多·范罗苏姆(Guido van Rossum),目前已经是使用频度特别高的开发语言. 主要应用领域: 云计算:云计算最火的语言,典型应用有Op ...
- SimpleDateFormat优化写法
在一个读取数据库数据导出到excel文件的例子当中,每次处理一个时间信息的时候,就需要创建一个SimpleDateFormat实例对象,然后再丢弃这个对象.大量的对象就这样被创建出来,占用大量的内存和 ...
- Django与多个数据库交互
定义数据库 在Django中使用多个数据库的第一步是告诉Django您将要使用的数据库服务器. 数据库可以有您选择的任何别名.但是,别名 default 有着特殊的意义.Django使用别名为 def ...
- leetcode-24-exercise
506. Relative Ranks 解题思路: 使用priority_queue.它在插入时会将数据按照由大到小的顺序插入,自然排序了.所以插入时考虑插入pair<nums[i],i> ...