1. 整整花了一天学习了LCA,tarjan的离线算法,就切了2个题。
  2. 第一题,给一棵树,一次查询,求LCA。2DFS+并查集,利用深度优先的特点,回溯的时候U和U的子孙的LCA是U,U和U的兄弟结点的子孙们的LCA是U的父亲,结合每次询问,

   3.   hdu2586,求无相无环有权图,求俩点距离(n<=40000,最短路必然TLE),转化树(任意取一点为根),双向边保存,链式前向星保存边和权,DfS,

先记录下每次询问,用链式前向星保存,双向保存,第(i+1)/2条边即为第i次询问(一次询问记为双向边),最后用一个数组ans[i][3]来记录,ans[i][0]:第i次询问起点,依次ans[i][1],终点,ans[i][2],他们的LCA。DIS【i】,表根到I的距离,最后:dis[u]+dis[v]-2*dis[lca[u,v]]即可。

#include<iostream>  //只求一次询问 1330,水
#include<vector>
#include<cstdio>
using namespace std;
vector<vector<int> >edge(10001);
int from,to;int n; bool mark=0;
int fa[10001];int visited[10001];
int find(int x){return fa[x]=(x==fa[x]?x:find(fa[x]));} //压缩路劲
void readin()
{
scanf("%d",&n); mark=0;
for(int i=0;i<=n;i++) //初始化
{
visited[i]=1;
fa[i]=i;
edge[i].clear();
}
int begin,end;
for(int i=0;i<n-1;i++)
{
scanf("%d%d",&begin,&end);
edge[begin].push_back(end);
visited[end]=0; //不是根,这样标记出根。
}
scanf("%d%d",&from,&to);
}
void tarjan(int u)
{ int len=edge[u].size();
for(int i=0;i<len;i++)
{
int v=edge[u][i];
if(visited[v]==0)
{
tarjan(v);
if(mark)return;
fa[v]=u; //合并之
}
}
visited[u]=1; //回溯时标记!!!!!这时候表明它和它的孩子都已经被标记,离线LCA这样。
if(u==from&&visited[to]) //查询询问,另外一个点是否已经访问,若访问了,以find(v)为他们的LCA
{
mark=1;
printf("%d\n",find(to));return;
}
if(u==to&&visited[from])
{
mark=1;
printf("%d\n",find(from));return;
}
}
int main()
{
int tcase;scanf("%d",&tcase);
while(tcase--)
{
readin();
for(int i=1;i<=n;i++)
{
if(visited[i]) //从根开使
{
visited[i]=0; //此处不忘把根标记回来!
tarjan(i);
break;
}
}
}
return 0;
}
#include<iostream>  //46MS(G++)     会了就水了,DIF 2
#include<vector> //o(n+q)
#include<cstdio>
using namespace std;
struct edges //边集
{
int pre,to,w;
};
struct querys //询问的边
{
int pre,to;
};
int n; int num_query;
int fa[40001];int visited[40001];
int dis[40001];int head[40001];int head2[40001];
int res[201][3];
vector<querys>que(401);
vector<edges>edge(80001);
int find(int x){return fa[x]=(x==fa[x]?x:find(fa[x]));} //并查集+压缩路径优化之
void readin()
{
scanf("%d%d",&n,&num_query);
for(int i=0;i<=n;i++) //初始化,
{
head[i]=head2[i]=-1;
visited[i]=0;
fa[i]=i;
dis[i]=0;
}
int begin,end,w;
for(int i=0;i<2*(n-1);i++) //读入边和询问,都双向读入
{
scanf("%d%d%d",&begin,&end,&w);
edge[i].to=end;
edge[i].w=w;
edge[i].pre=head[begin];
head[begin]=i;
i++;
edge[i].to=begin;
edge[i].w=w;
edge[i].pre=head[end];
head[end]=i;
}
for(int i=1;i<=2*num_query;i++) //询问也双向读入,防止只有一头的情况
{
scanf("%d%d",&begin,&end);
que[i].to=end;
que[i].pre=head2[begin];
head2[begin]=i;
i++;
que[i].to=begin;
que[i].pre=head2[end];
head2[end]=i;
}
}
void tarjan(int u,int father) //算法关键.
{
for(int i=head[u];i!=-1;i=edge[i].pre)
{
int v=edge[i].to;
if(visited[v]==0&&v!=father) //不回走(father) {
dis[v]=dis[u]+edge[i].w; //沿路记录长度
tarjan(v,u); //递归
fa[v]=u; //合并之
}
}
visited[u]=1; //回溯时标记(标记了说明该店已经有祖先fa[u]值),
for(int i=head2[u];i!=-1;i=que[i].pre) //遍历询问U的边,若有询问。用前向星可以记录询问的编号又能降低复杂度(直接访问U来走)
{
if(visited[que[i].to])
{
res[(i+1)/2][0]=u; //记录起点,终点,他们的LCA,询问的编号(次序)是(i+1)/2
res[(i+1)/2][1]=que[i].to;
res[(i+1)/2][2]= find(que[i].to);
}
}
}
int main()
{
int tcase;scanf("%d",&tcase);
while(tcase--)
{
readin();
tarjan(1,-1);
for(int i=1;i<=num_query;i++)
{
printf("%d\n",dis[res[i][0]]+dis[res[i][1]]-2*dis[res[i][2]]);
}
//printf("\n");
}
return 0;
}

poj1330+hdu2586 LCA离线算法的更多相关文章

  1. Tarjan的LCA离线算法

    LCA(Least Common Ancestors)是指树结构中两个结点的最低的公共祖先.而LCA算法则是用于求两个结点的LCA.当只需要求一对结点的LCA时,我们很容易可以利用递归算法在O(n)的 ...

  2. Closest Common Ancestors---poj1470(LCA+离线算法)

    题目链接:http://poj.org/problem?id=1470 题意是给出一颗树,q个查询,每个查询都是求出u和v的LCA:    以下是寻找LCA的预处理过程: void LCA(u){ f ...

  3. HDU 2874 Connections between cities(LCA离线算法实现)

    http://acm.hdu.edu.cn/showproblem.php?pid=2874 题意: 求两个城市之间的距离. 思路: LCA题,注意原图可能不连通. 如果不了解离线算法的话,可以看我之 ...

  4. LCA离线算法Tarjan详解

    离线算法也就是需要先把所有查询给保存下来,最后一次输出结果. 离线算法是基于并查集实现的,首先就是初始化P[i] = i. 接下来对于每个点进行dfs: ①首先判断是否有与该点有关的查询,如果当前该点 ...

  5. 距离LCA离线算法Tarjan + dfs + 并查集

    距离B - Distance in the Tree 还是普通的LCA但是要求的是两个节点之间的距离,学到了一些 一开始我想用带权并查集进行优化,但是LCA合并的过程晚于离线计算的过程,所以路径长度会 ...

  6. LCA离线算法Tarjan的模板

    hdu 2586:题意:输入n个点的n-1条边的树,m组询问任意点 a b之间的最短距离 思路:LCA中的Tarjan算法,RMQ还不会.. #include <stdio.h> #inc ...

  7. HDU 2874 LCA离线算法 tarjan算法

    给出N个点,M条边.Q次询问 Q次询问每两点之间的最短距离 典型LCA 问题   Marjan算法解 #include "stdio.h" #include "strin ...

  8. POJ1986 DistanceQueries 最近公共祖先LCA 离线算法Tarjan

    这道题与之前那两道模板题不同的是,路径有了权值,而且边是双向的,root已经给出来了,就是1,(这个地方如果还按之前那样来计算入度是会出错的.数据里会出现多个root...数据地址可以在poj的dis ...

  9. 1128 - Greatest Parent---LightOj(LCA+离线算法)

    题目链接:http://lightoj.com/volume_showproblem.php?problem=1128 给你一颗树,树的每个节点都有一个权值,树根是节点0,权值为1,树中每个节点的权值 ...

随机推荐

  1. Javaweb学习笔记8—DBUtils工具包

    今天来讲javaweb的第8阶段学习. DBUtils技术,DBUtils是我们操作数据库很常用的功能,虽然后期使用都是它的封装结果,但是也需要掌握. 老规矩,首先先用一张思维导图来展现今天的博客内容 ...

  2. SQLServer · 最佳实践 · SQL Server 2012 使用OFFSET分页遇到的问题

    1. 背景 最近有一个客户遇到一个奇怪的问题,以前使用ROW_NUMBER来分页结果是正确的,但是替换为SQL SERVER 2012的OFFSET...FETCH NEXT来分页出现了问题,因此,这 ...

  3. Java递归调用改成非递归

          在java语言中,使用递归调用时,如果过多的调用容易造成java.lang.StackOverflowError即栈溢出和程序执行过慢.这是一个潜在Bug和影响程序执行效率问题,需要谨慎使 ...

  4. MyEclipse 2015 安装到配置一站式备忘

    目录 h1 2121 h1 2121

  5. 递归的可视化(Fibonacci)

    递归的可视化 修改递归函数,使其能够显示打印出每次函数递归调用的形参的值. 每一级调用的输出都带有一级缩进,就是使得程序的输出清晰.有趣并且有含义. 思路 以斐波那契数列为例,假设n=5,递归的形参如 ...

  6. drawer 抽屉 弹框 在 modal的后面的解决方案

    drawer 抽屉 弹框 在 modal的后面的解决方案 方案1 在框内 弹出 <Drawer title="拍照" :transfer="false" ...

  7. Asp.Net Core 入门(二)——Startup.cs做了什么

    上篇介绍了Program.cs中Main做了什么,这篇我们来讨论下Startup.cs它又做了什么呢? 我们新建一个Asp.Net Core Mvc项目,先来开一下Startup的代码 public ...

  8. 【转】密码学 :CSP的概念

    转:[密码学]CSP的概念 CSP加密服务提供者(Cryptographic Service Provider)具有一下几个特点: CSP是真正执行密码运算的独立模块 物理上一个CSP由两部分组成:一 ...

  9. java_日期和时间

    1.System类中的currentTimeMillis:1970年1.1到现在的毫秒数 public class DateTest { public static void main(String[ ...

  10. ES6 第一章 let和const命令 具体参照http://es6.ruanyifeng.com

    1.let类似于var用用来定义变量 1)let没有预解析,不存在变量提升 // var 的情况 console.log(foo); // 输出undefined var foo = 2; // le ...