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. 新建cordova应用

    使用命令行(本例命令行均使用as或webstrom的命令行),在任意目录输入以下命令新建cordova应用 cordova create capp1 com.cesc.ewater.capp1 其中c ...

  2. 纯css滚动公告栏目

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  3. 迅为iMX6UL Cortex-A7架构单核ARM开发板接口介绍-支持定制

    支持商业级和工业级核心板 1. POWER 电源接口电源输入为 5V/2A+,给核心板提供 5V 电源,给底板供电.原理图部分如下图所示. 电源接口位置如下图所示. 2. SWITCH 电源开关轻触电 ...

  4. C-基础:C语言为什么不做数组下标越界检查

    //这段代码运行有可能不报错.]; ;i<;i++) { a[i]=i; } 1.为了提高运行效率,不检查数组下表越界,程序就可以跑得快.因为C语言并不是一个快速开发语言,它要求开发人员保证所有 ...

  5. Report Builder 打开报错

    提示:REP-0756:警告:找不到PL/SQL程序库’inv.pll’. 解决办法 处理:1.运行 ---输入REgedit ,在注册表内,查找注册表项Reports_path. 2.将下载的.pl ...

  6. [BZOJ2120]:数颜色(分块?)

    题目传送门 我感觉这种题没必要扯淡题目大意了,没啥用. 暴力过掉,擦了个边. 主要是讲一下这道题我用到的卡常. 首先,0,1标记我用的位运算,位运算符跑的要比正常的+,-,×,÷,true,false ...

  7. python TCP协议详解 三次握手四次挥手和11种状态

    11种状态解析 LISTEN  --------------------  等待从任何远端TCP 和端口的连接请求. SYN_SENT  ---------------  发送完一个连接请求后等待一个 ...

  8. 解决CSDN阅读全部需要登录的问题

    现在CSDN网站每次点击“阅读全部”的时候,必须要登录才能展开,很不方便.解决方法如下:点击F12打开开发者工具,点击Console,将下面两行代码粘贴进去即可: $("div.articl ...

  9. Java Web中的编码解析

    在springmvc工程web.xml中配置中文编码 <!-- 配置请求过滤器,编码格式设为UTF-8,避免中文乱码--> <filter> <filter-name&g ...

  10. 第三讲:post-processsing with vcs+ files

    1,dump wave  by system function $vcdpluson(level_number,module_instance,....|net_or_reg) $vcdplusoff ...