hdu2586 How far away

题目都翻译过了,只需用脑整理词序......

【题目描述】:

有N个房子在村里和一些双向道路连接。每一天,人们总是喜欢这样问“有多远,如果我想从房子到房子B”?通常很难回答。但幸运的是,在这个村的回答永远是独一无二的,由于道路建成的方式,这是一个独特的简单路径(“简单”意味着你不能访问一个地方两次)每两个房子之间。你的任务是回答这些好奇的人。

【输入】:

第一行是一个整数T(t<=10),表示测试用例的数目。
对于每个测试用例,在第一行有2个数n(2<=n<=40000)和M(1<=M<=200),房屋的数量和查询的数量。下面的n-1条线路每组三个数即,i,j,k分离不单空间,这意味着有一条连接房子房子我和J,长度为k(0<k<=40000)。房屋标记从1到n。
接下来的m行各有不同的整数i和j,你要回答我的房子i和房子J之间的距离

【输出】:

对于每一个测试案例,输出M行。每一行代表该查询的答案。输出一个平淡无奇的线每一个测试案例后。

【样例输入】:

 
 
2 3 2 1 2 10 3 1 15 1 2 2 3 2 2 1 2 100 1 2 2 1
【样例输出】:

 
 
10 25 100 100

题解Here!

此题为一道LCA问题,基本上可以算模板题。先找到询问两点(假设为u,v)的最近公共祖先,再计算距离。

距离:用dis[]记下根节点到任意节点的距离,则ans=dis[u]+dis[v]-2*dis[lca(u, v)];

上代码:

#include<iostream>
#include<algorithm>
#include<stdio.h>
#include<string.h>
#define MAXN 40010
#define MAXM 205
using namespace std;
typedef struct Node{
       int v,d;
       Node *next;
}node;
node *linka[MAXN],a[MAXN<<1],*linkb[MAXN],b[MAXN<<1];
int idx1,idx2,n,m;
int fa[MAXN],vis[MAXN],dis[MAXN],s[MAXM][3];
int find(int x){return fa[x]==x?x:fa[x]=find(fa[x]);}
inline int read(){
       int date=0,w=1;char c=0;
       while(c!='-'&&(c<'0'||c>'9'))c=getchar();
       if(c=='-'){w=-1;c=getchar();}
       while(c>='0'&&c<='9'){date=date*10+c-'0';c=getchar();}
       return date*w;
}
void add(int u,int v,int d,node c[],node *linkc[],int &idx){
     c[idx].v=v;
     c[idx].d=d;
     c[idx].next=linkc[u];
     linkc[u]=c+idx++;
     c[idx].v=u;
     c[idx].d=d;
     c[idx].next=linkc[v];
     linkc[v]=c+idx++;
}
void LCA(int u){
     vis[u]=1;
     fa[u]=u;
     for(node *p=linkb[u];p;p=p->next)
     if(vis[p->v])
     s[p->d][2]=find(p->v);
     for(node *p=linka[u];p;p=p->next)
     if(!vis[p->v]){
                   dis[p->v]=dis[u]+p->d;
                   LCA(p->v);
                   fa[p->v]=u;
                   }
}
int main(){
    int t,i,u,v,d;
    t=read();
    while(t--){
               n=read();m=read();
               idx1=idx2=0;
               memset(linka,0,sizeof(linka));
               memset(linkb,0,sizeof(linkb));
               for(i=1;i<=n;i++){
                                 u=read();v=read();d=read();
                                 add(u,v,d,a,linka,idx1);
                                 }
               for(i=1;i<=m;i++){
                                 u=read();v=read();
                                 add(u,v,i,b,linkb,idx2);
                                 s[i][0]=u;
                                 s[i][1]=v;
                                 }
               memset(vis,0,sizeof(vis));
               dis[1]=0;
               LCA(1);
               for(i=1;i<=m;i++)
               printf("%d\n",dis[s[i][0]]+dis[s[i][1]]-2*dis[s[i][2]]);
               }
    return 0;
}


  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值