UVA - 12487 Midnight Cowboy(LCA+思维)

https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&problem=3931

题意

一棵树,一个人从A节点出发,等可能选择到下一个点的路,有B、C两个结点,问先到C的概率。

分析

若A,B,C在一条链上,那么A到B,C的概率只与它们之间的长度有关。那么当A不在B,C所在的链上时,我们只需要找到进入这条链的入口,也就是B和C的公共祖先,通过这个公共祖先就能从A走到链上,而这之前的路对于B和C来说都是等价的,因此无关。

 

#include<iostream>
#include<cmath>
#include<cstring>
#include<queue>
#include<vector>
#include<cstdio>
#include<algorithm>
#include<map>
#include<set>
#define rep(i,e) for(int i=0;i<(e);i++)
#define rep1(i,e) for(int i=1;i<=(e);i++)
#define repx(i,x,e) for(int i=(x);i<=(e);i++)
#define X first
#define Y second
#define PB push_back
#define MP make_pair
#define mset(var,val) memset(var,val,sizeof(var))
#define scd(a) scanf("%d",&a)
#define scdd(a,b) scanf("%d%d",&a,&b)
#define scddd(a,b,c) scanf("%d%d%d",&a,&b,&c)
#define pd(a) printf("%d\n",a)
#define scl(a) scanf("%lld",&a)
#define scll(a,b) scanf("%lld%lld",&a,&b)
#define sclll(a,b,c) scanf("%lld%lld%lld",&a,&b,&c)
#define IOS ios::sync_with_stdio(false);cin.tie(0)
using namespace std;
typedef long long ll;
template <class T>
void test(T a){cout<<a<<endl;}
template <class T,class T2>
void test(T a,T2 b){cout<<a<<" "<<b<<endl;}
template <class T,class T2,class T3>
void test(T a,T2 b,T3 c){cout<<a<<" "<<b<<" "<<c<<endl;}
const int inf = 0x3f3f3f3f;
const ll INF = 0x3f3f3f3f3f3f3f3fll;
const ll mod = 1e9+7;
int T;
void testcase(){
    printf("Case %d: ",++T);
}
const int MAXN = 500;
const int MAXM = 30;

struct Edge{
    int to,nxt;
}e[MAXN*2];
int head[MAXN],tot;
void addEdge(int u,int v){
    e[tot].to=v;
    e[tot].nxt=head[u];
    head[u]=tot++;
}
void init(){
    tot=0;
    mset(head,-1);
}
int fa[MAXN][20];
int deg[MAXN];
void BFS(int rt){
    queue<int> que;
    deg[rt]=0;
    fa[rt][0]=rt;
    que.push(rt);
    while(!que.empty()){
        int tmp = que.front();
        que.pop();
        for(int i=1;i<20;i++) fa[tmp][i]=fa[fa[tmp][i-1]][i-1];
        for(int i=head[tmp];~i;i=e[i].nxt){
            int v = e[i].to;
            if(v==fa[tmp][0]) continue;
            deg[v]=deg[tmp]+1;
            fa[v][0]=tmp;
            que.push(v);
        }
    }
}


int LCA(int u,int v){
    if(deg[u]>deg[v]) swap(u,v);
    int hu = deg[u],hv=deg[v];
    int tu = u,tv=v;
    for(int det=hv-hu,i=0;det;det>>=1,i++){
        if(det&1) tv=fa[tv][i];
    }
    if(tu==tv) return tu;
    for(int i=19;i>=0;i--){
        if(fa[tu][i]==fa[tv][i]){
            continue;
        }
        tu=fa[tu][i];
        tv=fa[tv][i];
    }
    return fa[tu][0];
}

int main() {
#ifdef LOCAL
    freopen("in.txt","r",stdin);
#endif // LOCAL
    int A,B,C,N;
    while(~scanf("%d%d%d%d",&N,&A,&B,&C)){
        init();
        int u,v;
        for(int i=1;i<N;i++){
            scdd(u,v);
            addEdge(u,v);
            addEdge(v,u);
        }
        BFS(A);
        int lca =LCA(B,C);
        int c=deg[C]-deg[lca];
        int b=deg[B]-deg[lca];
        double ans=0;
        if(c==0) ans=0;
        else if(b==0) ans=1;
        else ans=1.0*c/(c+b);
        printf("%.6f\n",ans);
    }
    return 0;
}

 

posted @ 2018-07-12 22:17  litos  阅读(126)  评论(0编辑  收藏  举报