LCA(离线算法)
CD操作
Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65535/32768 K (Java/Others)
Total Submission(s): 1010 Accepted Submission(s): 275
这里我们简化一下问题,假设只有一个根目录,CD操作也只有两种方式:
1. CD 当前目录名\...\目标目录名 (中间可以包含若干目录,保证目标目录通过绝对路径可达)
2. CD .. (返回当前目录的上级目录)
现在给出当前目录和一个目标目录,请问最少需要几次CD操作才能将当前目录变成目标目录?
每个样例首先一行是两个整数N和M(1<=N,M<=100000),表示有N个目录和M个询问;
接下来N-1行每行两个目录名A B(目录名是只含有数字或字母,长度小于40的字符串),表示A的父目录是B。
最后M行每行两个目录名A B,表示询问将当前目录从A变成B最少要多少次CD操作。
数据保证合法,一定存在一个根目录,每个目录都能从根目录访问到。
2
3 1
B A
C A
B C 3 2
B A
C B
A C
C A
2
1
2
方法一:
1700ms
#include"stdio.h"
#include"string.h"
#include"stdlib.h"
#define M 100009
#include"string"
#include"map"
#include"iostream"
using namespace std;
typedef struct st
{
int u,v,next,w;
}E[M*3];
E edge,edge1;
int dis[M],head[M],head1[M],t,t1,use[M],in[M];
int f[M];
int finde(int x)
{
if(x!=f[x])
f[x]=finde(f[x]);
return f[x];
}
void make(int u,int v)
{
int x=finde(u);
int y=finde(v);
if(x!=y)
f[x]=y;
}
void init()
{
t=t1=0;
memset(head,-1,sizeof(head));
memset(head1,-1,sizeof(head1));
}
void add(int u,int v)
{
edge[t].u=u;
edge[t].v=v;
edge[t].next=head[u];
head[u]=t++;
}
void add1(int u,int v,int w)
{
edge1[t1].u=u;
edge1[t1].v=v;
edge1[t1].w=w;
edge1[t1].next=head1[u];
head1[u]=t1++;
}
void dfs(int u)
{
use[u]=1;
int i;
for(i=head[u];i!=-1;i=edge[i].next)
{
int v=edge[i].v;
if(!use[v])
{
dis[v]=dis[u]+1;
dfs(v);
f[v]=u;
make(u,v);
}
}
for(i=head1[u];i!=-1;i=edge1[i].next)
{
int v=edge1[i].v;
if(use[v])
{
edge1[i].w=edge1[i^1].w=f[finde(v)];
}
}
}
int main()
{
int T,i,m,n,x,y;
char ch1[60],ch2[60];
scanf("%d",&T);
while(T--)
{
scanf("%d%d",&n,&m);
map<string,int>mp;
int k=1;
init();
memset(in,0,sizeof(in));
for(i=1;i<n;i++)
{
scanf("%s%s",ch1,ch2);
if(mp[ch1]==0)
{
x=k;
mp[ch1]=k++;
}
else
x=mp[ch1];
if(mp[ch2]==0)
{
y=k;
mp[ch2]=k++;
}
else
y=mp[ch2];
add(y,x);
in[x]++;
}
while(m--)
{
scanf("%s%s",ch1,ch2);
add1(mp[ch1],mp[ch2],0);
add1(mp[ch2],mp[ch1],0);
}
memset(use,0,sizeof(use));
memset(dis,0,sizeof(dis));
for(i=1;i<=n;i++)
f[i]=i;
for(i=1;i<=n;i++)
{
if(!in[i])
dfs(i);
}
for(i=0;i<t1;i+=2)
{
int u=edge1[i].u;
int v=edge1[i].v;
int mid=edge1[i].w;
int p;
if(v==mid)
p=0;
else
p=1;
printf("%d\n",dis[u]-dis[mid]+p);
}
}
}
方法二:
2700ms
#include"stdio.h"
#include"string.h"
#include"map"
#include"iostream"
#include"queue"
using namespace std;
#define M 100006
int dis[M];
int pre[M];
int rank[M],use[M],t,head[M];
int targan(int a,int b)
{
if(a==b)
return a;
else if(rank[a]>rank[b])
return targan(pre[a],b);
else
return targan(a,pre[b]);
}
struct st
{
int u,v,next;
}edge[M*3];
void init()
{
t=0;
memset(head,-1,sizeof(head)); }
void add(int u,int v)
{
edge[t].u=u;
edge[t].v=v;
edge[t].next=head[u];
head[u]=t++;
}
void bfs(int s)
{
queue<int>q;
memset(use,0,sizeof(use));
use[s]=1;
memset(rank,0,sizeof(rank));
q.push(s);
while(!q.empty())
{
int u=q.front();
q.pop();
for(int i=head[u];i!=-1;i=edge[i].next)
{
int v=edge[i].v;
if(!use[v])
{
use[v]=1;
rank[v]=rank[u]+1;
q.push(v);
}
}
}
}
int main()
{
int w;
scanf("%d",&w);
while(w--)
{
map<string,int>mp;
int n,m,i;
char ch1[444],ch2[444];
init();
scanf("%d%d",&n,&m);
for(i=1;i<=n;i++)
pre[i]=i;
int t=1;
for(i=1;i<n;i++)
{
scanf("%s%s",ch1,ch2);
if(!mp[ch1])
mp[ch1]=t++;
if(!mp[ch2])
mp[ch2]=t++;
pre[mp[ch1]]=mp[ch2];
dis[mp[ch1]]=1;
add(mp[ch2],mp[ch1]);
}
int tep=-1;
for(i=1;i<=n;i++)
if(pre[i]==i)
tep=i;
bfs(tep);
while(m--)
{
scanf("%s%s",ch1,ch2);
int ans=targan(mp[ch1],mp[ch2]);
if(ans!=mp[ch2])
printf("%d\n",rank[mp[ch1]]-rank[ans]+1);
else
printf("%d\n",rank[mp[ch1]]-rank[ans]);
}
}
return 0; }
LCA(离线算法)的更多相关文章
- Tarjan的LCA离线算法
LCA(Least Common Ancestors)是指树结构中两个结点的最低的公共祖先.而LCA算法则是用于求两个结点的LCA.当只需要求一对结点的LCA时,我们很容易可以利用递归算法在O(n)的 ...
- Closest Common Ancestors---poj1470(LCA+离线算法)
题目链接:http://poj.org/problem?id=1470 题意是给出一颗树,q个查询,每个查询都是求出u和v的LCA: 以下是寻找LCA的预处理过程: void LCA(u){ f ...
- HDU 2874 Connections between cities(LCA离线算法实现)
http://acm.hdu.edu.cn/showproblem.php?pid=2874 题意: 求两个城市之间的距离. 思路: LCA题,注意原图可能不连通. 如果不了解离线算法的话,可以看我之 ...
- LCA离线算法Tarjan详解
离线算法也就是需要先把所有查询给保存下来,最后一次输出结果. 离线算法是基于并查集实现的,首先就是初始化P[i] = i. 接下来对于每个点进行dfs: ①首先判断是否有与该点有关的查询,如果当前该点 ...
- poj1330+hdu2586 LCA离线算法
整整花了一天学习了LCA,tarjan的离线算法,就切了2个题. 第一题,给一棵树,一次查询,求LCA.2DFS+并查集,利用深度优先的特点,回溯的时候U和U的子孙的LCA是U,U和U的兄弟结点的子孙 ...
- 距离LCA离线算法Tarjan + dfs + 并查集
距离B - Distance in the Tree 还是普通的LCA但是要求的是两个节点之间的距离,学到了一些 一开始我想用带权并查集进行优化,但是LCA合并的过程晚于离线计算的过程,所以路径长度会 ...
- LCA离线算法Tarjan的模板
hdu 2586:题意:输入n个点的n-1条边的树,m组询问任意点 a b之间的最短距离 思路:LCA中的Tarjan算法,RMQ还不会.. #include <stdio.h> #inc ...
- HDU 2874 LCA离线算法 tarjan算法
给出N个点,M条边.Q次询问 Q次询问每两点之间的最短距离 典型LCA 问题 Marjan算法解 #include "stdio.h" #include "strin ...
- POJ1986 DistanceQueries 最近公共祖先LCA 离线算法Tarjan
这道题与之前那两道模板题不同的是,路径有了权值,而且边是双向的,root已经给出来了,就是1,(这个地方如果还按之前那样来计算入度是会出错的.数据里会出现多个root...数据地址可以在poj的dis ...
- 1128 - Greatest Parent---LightOj(LCA+离线算法)
题目链接:http://lightoj.com/volume_showproblem.php?problem=1128 给你一颗树,树的每个节点都有一个权值,树根是节点0,权值为1,树中每个节点的权值 ...
随机推荐
- there are 0 datanode.....
当时执行hive的导入数据load data inpath "XXXX" into table.....的时候发现总是导不进去,最后试了下简单的从Linux 到 HDFS上传文件 ...
- android 监听Home键
/** * Home 键监听,当按下Home键时,系统会发出action为Intent.ACTION_CLOSE_SYSTEM_DIALOGS的BroadcastReceiver * 在程序里动态注册 ...
- 关于Cocos2d-x中文乱码问题的解决
方法一: 1.首先,复制下面的代码,创建一个icov,h的头文件,并放在项目目录下 #include "stdlib.h"#include "string.h" ...
- 【转】【Asp.Net】ASP.Net Response.ContentType 详细列表
不同的ContentType 会影响客户端所看到的效果.默认的ContentType为 text/html 也就是网页格式. 代码如: <% response.ContentType =&quo ...
- php一些常规动态设置与获取
error_reporting(E_ALL); ini_set('display_errors', TRUE); ini_set('display_startup_errors', TRUE);ini ...
- 新机器,分区为NTFS, 安装 Windows XP、Windows Server 2003 时蓝屏问题,修改为 FAT32 即可
现象:安装刚刚开始就会蓝屏:Ghost版本的也无法正常开机. 原因:硬盘引起,通常是主板的RAID.或STAT的设置引起????? 最直接的原因是安装所在的分区的文件系统格式不正确,为NTFS 解决: ...
- 解决导入protobuf源代码Unity报错的问题
将源代码导入Assets目录后, unity引擎会出现以下报错: 解决办法: 在 unity项目Assets目录中创建smcs.rsp文件,内容为-unsafe,其作用为可编译不安全代码. 然 ...
- CSS background 之设置图片为背景技巧
首先先来看看background有那些值: 可以按顺序设置如下属性(可点击进入相应的css手册查看使用):background-color 背景颜色background-image 背景图片backg ...
- [dubbo] dubbo 基础使用
Dubbo是一个分布式服务框架,致力于提供高性能和透明化的RPC远程服务调用方案,以及SOA服务治理方案.简单的说,dubbo就是个服务框架,如果没有分布式的需求,其实是不需要用的,只有在分布式的时候 ...
- ADO.net方法
using System; using System.Collections.Generic; using System.Data; using System.Data.SqlClient; usin ...