lca最短公共祖先模板(hdu2586)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2586

#include<iostream>
#include<cstdio>
#include<cmath>
#include<queue>
#include<vector>
#include<string.h>
#include<cstring>
#include<algorithm>
#include<set>
#include<map>
#include<fstream>
#include<cstdlib>
#include<ctime>
#include<list>
#include<climits>
#include<bitset>
using namespace std;
#define fio ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);
#define fopen freopen("input.txt", "r", stdin);freopen("output.txt", "w", stdout);
#define scd(a) scanf("%d",&a)
#define scf(a) scanf("%lf",&a)
#define scl(a) scanf("%lld",&a)
#define sci(a) scanf("%I64d",&a)
#define scs(a) scanf("%s",a)
#define left asfdasdasdfasdfsdfasfsdfasfdas1
#define tan asfdasdasdfasdfasfdfasfsdfasfdas
typedef long long ll;
typedef unsigned int un;
const int desll[4][2]={{0,1},{0,-1},{1,0},{-1,0}};
const ll mod=1e9+7;
const int maxn=4e4+7;
const int maxm=3e8+7;
const double eps=1e-4;
int m,n;

struct node
{
    int b,nex,c;
}no[maxn*2];
int head[maxn],sz=0;
int add(int a,int b,int c){
    no[sz].b=b;
    no[sz].c=c;
    no[sz].nex=head[a];
    head[a]=sz++;
}
int father[maxn],tan[maxn*2][30],dep[maxn];
int ar[maxn*2],le,in[maxn];
ll dis[maxn];
void dfs1(int u,int pre){
    for(int i=head[u];i!=-1;i=no[i].nex){
        int v=no[i].b;
        if(v==pre)continue;
        dis[v]=dis[u]+no[i].c;
        father[v]=u;
        dep[v]=dep[u]+1;
        ar[le++]=u;
        dfs1(v,u);
    }
    in[u]=le;
    ar[le++]=u;
}
void init()
{
    memset(tan,0,sizeof(tan));
    for(int i=0;i<le;i++){
        tan[i][0]=ar[i];
    }
    for(int j=1;j<20;j++){
        for(int i=le-1;i>=0;i--){
            if(i+(1<<j)-1>=le)continue;
            int x=tan[i][j-1],y=tan[i+(1<<(j-1))][j-1];
            if(dep[x]<dep[y])tan[i][j]=x;
            else tan[i][j]=y;
            //cout<<i<<" "<<j<<" "<<tan[i][j]<<endl;
        }
    }
}
int sameFather(int a,int b)
{
    a=in[a];
    b=in[b];
    if(a>b)swap(a,b);
    int res=floor(log(b-a+1)/log(2));
    int x=tan[a][res],y=tan[b-(1<<res)+1][res];
    if(dep[x]<dep[y])return x;
    else return y;
}

int main()
{
    int t;scd(t);
    while(t--){
        scd(n);scd(m);
        memset(head,-1,sizeof(head));sz=0;
        for(int i=1;i<n;i++){
            int a,b,c;
            scd(a);scd(b);scd(c);
            add(a,b,c);
            add(b,a,c);
        }
        memset(dis,0,sizeof(dis));
        memset(dep,0,sizeof(dep));
        dep[1]=1;
        dfs1(1,0);
        father[1]=0;
        init();
        for(int i=0;i<m;i++){
            int a,b;
            scd(a);scd(b);
            int x=sameFather(a,b);
            printf("%d\n",dis[a]+dis[b]-2*dis[x]);
        }
    }
    return 0;
}

原文地址:https://www.cnblogs.com/wa007/p/9697104.html

时间: 2024-11-14 18:23:42

lca最短公共祖先模板(hdu2586)的相关文章

笔记:LCA最近公共祖先 Tarjan(离线)算法

LCA最近公共祖先 Tarjan他贱(离线)算法的基本思路及其算法实现 本文是网络资料整理或部分转载或部分原创,参考文章如下: https://www.cnblogs.com/JVxie/p/4854719.html http://blog.csdn.net/ywcpig/article/details/52336496 https://baike.baidu.com/item/最近公共祖先/8918834?fr=aladdin 最近公共祖先简称LCA(Lowest Common Ancesto

lca 最近公共祖先

http://poj.org/problem?id=1330 1 #include<cstdio> 2 #include<cstring> 3 #include<algorithm> 4 #define mt(a,b) memset(a,b,sizeof(a)) 5 using namespace std; 6 const int inf=0x3f3f3f3f; 7 class LCA { ///最近公共祖先 build_o(n*logn) query_o(1) 8 t

Codeforces Round #362 (Div. 2) C. Lorenzo Von Matterhorn LCA(最近公共祖先)

C. Lorenzo Von Matterhorn time limit per test 1 second memory limit per test 256 megabytes input standard input output standard output Barney lives in NYC. NYC has infinite number of intersections numbered with positive integers starting from 1. Ther

POJ 1330 LCA最近公共祖先 离线tarjan算法

题意要求一棵树上,两个点的最近公共祖先 即LCA 现学了一下LCA-Tarjan算法,还挺好理解的,这是个离线的算法,先把询问存贮起来,在一遍dfs过程中,找到了对应的询问点,即可输出 原理用了并查集和dfs染色,先dfs到底层开始往上回溯,边并查集合并 一边染色,这样只要询问的两个点均被染色了,就可以输出当前并查集的最高父亲一定是LCA,因为我是从底层层层往上DSU和染色的,要么没被染色,被染色之后,肯定就是当前节点是最近的 #include <iostream> #include <

连通分量模板:tarjan: 求割点 &amp;&amp; 桥 &amp;&amp; 缩点 &amp;&amp; 强连通分量 &amp;&amp; 双连通分量 &amp;&amp; LCA(最近公共祖先)

PS:摘自一不知名的来自大神. 1.割点:若删掉某点后,原连通图分裂为多个子图,则称该点为割点. 2.割点集合:在一个无向连通图中,如果有一个顶点集合,删除这个顶点集合,以及这个集合中所有顶点相关联的边以后,原图变成多个连通块,就称这个点集为割点集合. 3.点连通度:最小割点集合中的顶点数. 4.割边(桥):删掉它之后,图必然会分裂为两个或两个以上的子图. 5.割边集合:如果有一个边集合,删除这个边集合以后,原图变成多个连通块,就称这个点集为割边集合. 6.边连通度:一个图的边连通度的定义为,最

LCA最近公共祖先

Nearest Common Ancestors Description A rooted tree is a well-known data structure in computer science and engineering. An example is shown below: In the figure, each node is labeled with an integer from {1, 2,...,16}. Node 8 is the root of the tree.

D5 LCA 最近公共祖先

第一题: POJ 1330 Nearest Common Ancestors POJ 1330 这个题可不是以1为根节点,不看题就会一直wa呀: 加一个找根节点的措施: #include<algorithm> #include<bitset> #include<cctype> #include<cerrno> #include<clocale> #include<cmath> #include<complex> #incl

[转]LCA 最近公共祖先

原文传送门orzJVxie Tarjan(离线)算法的基本思路及其算法实现 首先是最近公共祖先的概念(什么是最近公共祖先?): 在一棵没有环的树上,每个节点肯定有其父亲节点和祖先节点,而最近公共祖先,就是两个节点在这棵树上深度最大的公共的祖先节点. 换句话说,就是两个点在这棵树上距离最近的公共祖先节点. 所以LCA主要是用来处理当两个点仅有唯一一条确定的最短路径时的路径. 有人可能会问:那他本身或者其父亲节点是否可以作为祖先节点呢? 答案是肯定的,很简单,按照人的亲戚观念来说,你的父亲也是你的祖

(转)Tarjan应用:求割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)

本文转载自:http://hi.baidu.com/lydrainbowcat/item/f8a5ac223e092b52c28d591c 作者提示:在阅读本文之前,请确保您已经理解并掌握了基本的Tarjan算法,不会的请到http://hi.baidu.com/lydrainbowcat/blog/item/42a6862489c98820c89559f3.html阅读.   基本概念:   1.割点:若删掉某点后,原连通图分裂为多个子图,则称该点为割点. 2.割点集合:在一个无向连通图中,如