zoukankan      html  css  js  c++  java
  • 2019牛客暑期多校训练营(第六场) H:Train Driver (最短路+概率)

    题意:给定无向图,Alice在A集合选一个点,Bob在B集合选一个点,CXK在全集里选择一个点。 然后问“三人到某一点集合打篮球的最小距离”的期望。

    思路:做过一个裸题,就是给定三人位置,问去哪里集合距离代价最小。 那题就是三个点跑三次SPFA,就可以更新答案了。而此题有一个Cxk,非常的头疼,然而注意到边权为1,tm的不是直接BFS扩展就可以了吗。枚举Alice和Bob的位置,然后dis[i]=disA[i]+disB[j],然后就可以扩展了。 注意不要带log就可以过这题了,排序可以用基数排序,然后维护两个单调队列,每次取小的一个队首进行扩展。

    #include<bits/stdc++.h>
    #define ll long long
    #define rep(i,a,b) for(int i=a;i<=b;i++)
    using namespace std;
    const int maxn=200010;
    const int inf=1e9;
    int disA[21][maxn],disB[21][maxn],a[maxn],b[maxn];
    int Laxt[maxn],Next[maxn],To[maxn],cnt,N; ll sum;
    void add(int u,int v)
    {
        Next[++cnt]=Laxt[u]; Laxt[u]=cnt; To[cnt]=v;
    }
    void BFS(int dis[],int st)
    {
        rep(i,1,N) dis[i]=inf; dis[st]=0;
        queue<int>q; q.push(st);
        while(!q.empty()){
            int u=q.front(); q.pop();
            for(int i=Laxt[u];i;i=Next[i]){
                if(dis[To[i]]==inf){
                    dis[To[i]]=dis[u]+1;
                    q.push(To[i]);
                }
            }
        }
    }
    int num[maxn],c[maxn],dis[maxn];
    void solve() //基数排序+两个单调队列
    {
        rep(i,0,N+N) num[i]=0;
        rep(i,0,N) num[dis[i]]++,d[i]=0;
        rep(i,1,N+N) num[i]+=num[i-1];
        rep(i,1,N) c[num[dis[i]]--]=i;
        queue<int>q1,q2;
        rep(i,1,N) q1.push(c[i]);
        while(!q1.empty()||!q2.empty()){
            if(q2.empty()||(!q1.empty()&&!q2.empty()&&dis[q1.front()]<dis[q2.front()])) {
                int u=q1.front(); q1.pop();
                for(int i=Laxt[u];i;i=Next[i]) {
                    if(dis[To[i]]>dis[u]+1) {
                        dis[To[i]]=dis[u]+1;
                        q2.push(To[i]);
                    }
                }
            }
            else {
                int u=q2.front(); q2.pop();
                for(int i=Laxt[u];i;i=Next[i]) {
                    if(dis[To[i]]>dis[u]+1) {
                        dis[To[i]]=dis[u]+1;
                        q2.push(To[i]);
                    }
                }
            }
        }
        rep(i,1,N) sum+=dis[i];
    }
    int main()
    {
        int C=0,T,M,A,B,u,v;
        scanf("%d",&T);
        while(T--){
            scanf("%d%d",&N,&M);
            rep(i,1,N) Laxt[i]=0; cnt=0;
            rep(i,1,M){
               scanf("%d%d",&u,&v);
               add(u,v); add(v,u);
            }
            scanf("%d",&A); rep(i,1,A) scanf("%d",&a[i]);
            scanf("%d",&B); rep(i,1,B) scanf("%d",&b[i]);
            rep(i,1,A) BFS(disA[i],a[i]);
            rep(i,1,B) BFS(disB[i],b[i]);
            sum=0;
            rep(i,1,A)
             rep(j,1,B){
                 rep(k,1,N) dis[k]=disA[i][k]+disB[j][k];
                 solve();
            }
            ll ans=1LL*A*B*N;
            ll g=__gcd(ans,sum);
            printf("Case #%d: ",++C);
            if(g==sum) printf("%lld
    ",sum/g);
            else printf("%lld/%lld
    ",sum/g,ans/g);
        }
        return 0;
    }
  • 相关阅读:
    从留言簿开始,学习MonoRail MVC(三)
    从留言簿开始,学习MonoRail MVC(二)
    程序集版本最后一位使用SVN版本号的自动生成方法
    如何让.Net控件在设计时InitializeComponent()中不生成相关代码
    [收藏]Web Services and C# Enums
    从留言簿开始,学习MonoRail MVC(一)
    .Net控制USB设备相关内容
    .Net 2.0ListView控件在Windows 2000和Windows XP上的差异
    基于高德地图Windows Phone API 快速开发地图相关APP(二)
    android map api v2 示例 步骤及问题
  • 原文地址:https://www.cnblogs.com/hua-dong/p/11306550.html
Copyright © 2011-2022 走看看