zoukankan      html  css  js  c++  java
  • [poj1986]Distance Queries(LCA)

    解题关键:LCA模板题

    复杂度:$O(nlog n)$

      1 #pragma comment(linker, "/STACK:1024000000,1024000000")
      2 #include<cstdio>
      3 #include<cstring>
      4 #include<algorithm>
      5 #include<cstdlib>
      6 #include<cmath>
      7 #include<iostream>
      8 typedef long long ll;
      9 using namespace std;  
     10 const int maxn=81000; 
     11 const int maxm=25;  
     12 int _pow[maxm],m,n;  
     13 int head[maxn],tot;  
     14 int ver[maxn*2],depth[maxn*2],first[maxn],rmq[maxn*2][maxm],id;//5个数组,注意哪个需要乘2 
     15 ll dis[maxn];
     16 inline int read(){
     17     char k=0;char ls;ls=getchar();for(;ls<'0'||ls>'9';k=ls,ls=getchar());
     18     int x=0;for(;ls>='0'&&ls<='9';ls=getchar())x=(x<<3)+(x<<1)+ls-'0';
     19     if(k=='-')x=0-x;return x;
     20 }
     21 
     22 struct edge{
     23     int to,w,nxt;  
     24 }e[maxn*2];//链式前向星建树 
     25 
     26 void init(){
     27     memset(head,-1,sizeof head);  
     28     tot=0;
     29     id=0;
     30 }
     31 
     32 void add_edge(int u,int v,int w){  
     33     e[tot].to=v;
     34     e[tot].w=w;
     35     e[tot].nxt=head[u];
     36     head[u]=tot++;
     37 }
     38   
     39 void dfs(int u,int fa,int dep){
     40     ver[++id]=u;//第i个访问到的结点编号 
     41     depth[id]=dep;//第i个访问到的结点深度  
     42     first[u]=id;
     43     for(int i=head[u];i!=-1;i=e[i].nxt){
     44         int v=e[i].to;
     45         int w=e[i].w;
     46         if(v==fa) continue;
     47         dis[v]=dis[u]+w;//dis是先序遍历求 
     48         dfs(v,u,dep+1);
     49         ver[++id]=u;//后序遍历,再次访问父节点 
     50         depth[id]=dep;
     51     }
     52 } 
     53   
     54 void rmq_init(int n){  
     55     int k=int(log(n)/log(2));  
     56     for(int i=1;i<=n;++i)    rmq[i][0]=i;  
     57     for(int j=1;j<=k;++j){  
     58         for(int i=1;i+_pow[j]-1<=n;++i){//因为存的是索引 
     59             int a=rmq[i][j-1],b=rmq[i+_pow[j-1]][j-1];  
     60             if(depth[a]<depth[b])    rmq[i][j]=a;  
     61             else  rmq[i][j]=b;  
     62         }
     63     }
     64 }
     65   
     66 int rmq_query(int l,int r){
     67     int k=int(log(r-l+1.0)/log(2.0));  
     68     int a=rmq[l][k],b=rmq[r-_pow[k]+1][k];  
     69     if(depth[a]<depth[b])  return a;  
     70     else  return b;
     71 }//返回的依然是索引 
     72   
     73 int LCA(int u,int v){
     74     int x=first[u],y=first[v];  
     75     if(x>y)swap(x,y);  
     76     int res=rmq_query(x,y);  
     77     return ver[res]; 
     78 }
     79 
     80 int main(){
     81     for(int i=0;i<maxm;++i)    _pow[i]=1<<i; //预处理2^n 
     82     int t,a,b,c;
     83     init();
     84     n=read(),m=read();
     85     for(int i=0;i<m;++i){
     86         scanf("%d%d%d %*c",&a,&b,&c);
     87         add_edge(a,b,c);
     88         add_edge(b,a,c);
     89     }
     90     dfs(1,-1,0);
     91     rmq_init(2*n-1);
     92     int m;
     93     m=read();
     94     for(int i=0;i<m;++i){
     95         a=read();b=read();
     96         int ans=LCA(a,b);
     97         printf("%lld
    ",dis[a]+dis[b]-2*dis[ans]);
     98     }
     99     
    100     return 0;
    101 }
  • 相关阅读:
    【GO】文件二进制读取
    【JAVA】IO FileInputStream 读取二进制文件
    【JAVA】Maven 常用命令
    【JAVA】java8 function apply() compose() andThen()
    【Java】Maven resoucese 目录文件获取
    【数据结构】线索二叉树
    【Java】Java中的null作为方法参数是被当做值传递
    【Python】《Effective Python》 读书笔记 (一)
    【Python】一个try/except/else/finally 组合使用的例子
    【数据结构】二叉树的创建与遍历
  • 原文地址:https://www.cnblogs.com/elpsycongroo/p/7469609.html
Copyright © 2011-2022 走看看