zoukankan      html  css  js  c++  java
  • [UVA1494] Qin Shi Huang's National Road System

    题目

    戳这里

    题解

    从今天起我要改邪归正,好好刷题准备联赛!

    这是一道经典的最小生成树题目。
    枚举每一条边作为道士要修的路,求出包含这条边的最小生成树。

    先求出原图的最小生成树。
    如果要删的边在最小生成树上,那仍是原来那个最小生成树。
    如果不在,便要把这条边加进去。类似次小生成树,删除原最小生成树中这两点间唯一路径上边权最大的边,并把这条边加进去。

    我们要预处理最小生成树上两点间路径上边权最大的边,设它的边权为 (f[u][v])
    在求最小生成树的同时 借助父节点更新某一节点到已有生成树中其它节点的 (f) 值,复杂度 (O(n^2))

    代码

    #include<cstdio>
    #include<iostream>
    #include<algorithm>
    #include<cmath>
    #include<queue>
    
    using namespace std;
    
    const int N = 1005;
    typedef double db;
    typedef pair<double,int> P;
    
    int w[N],vis[N];
    db mp[N][N],a[N],b[N];
    int n;
    
    priority_queue< P, vector<P>, greater<P> > que;
    db d[N],f[N][N],S;
    int fa[N];
    void prim(){
        d[1]=0; vis[1]=1; 
        for(int i=2;i<=n;i++) 
            d[i]=mp[1][i],fa[i]=1,que.push(P(d[i],i));
        while(!que.empty()){
            int u=que.top().second;
            que.pop();
            if(vis[u]) continue;
            for(int i=1;i<=n;i++)
                if(vis[i]) {
                    if(i==fa[u]) f[i][u]=f[u][i]=mp[u][i];
                    else f[i][u]=f[u][i]=max(mp[u][fa[u]],f[fa[u]][i]);
                }
            vis[u]=1; d[u]=0;
            S+=mp[u][fa[u]];
            for(int v=1;v<=n;v++){
                if(v==u) continue;
                if(d[v]>mp[u][v]) 
                    fa[v]=u,d[v]=mp[u][v],que.push(P(d[v],v)); 
            }
        }
        for(int i=1;i<=n;i++) vis[i]=0;
    }
    
    int main()
    {
        int T;
        scanf("%d",&T);
        while(T--){
            scanf("%d",&n);
            for(int i=1;i<=n;i++)
                scanf("%lf%lf%d",&a[i],&b[i],&w[i]);
            for(int i=1;i<n;i++)
                for(int j=i+1;j<=n;j++)
                    mp[i][j]=mp[j][i]=sqrt((b[i]-b[j])*(b[i]-b[j])+(a[i]-a[j])*(a[i]-a[j]));
            S=0;
            prim();
            db ans=0.0;
            for(int i=1;i<n;i++)
                for(int j=i+1;j<=n;j++)
                    ans=max(ans,(w[i]*1.0+w[j]*1.0)/(S-f[i][j]));
            printf("%.2lf
    ",ans);
        }
        
        return 0;
    }
    
    既然选择了远方,便只顾风雨兼程
  • 相关阅读:
    js 进阶笔记
    Move Zeroes
    笔记
    前端笔记
    PAI-AutoLearning 图像分类使用教程
    Redis Key过期通知
    Ubuntu 安装配置 JDK+Tomcat+Nginx
    Linux常用指令总结
    使用Openssl创建证书
    Python 操作Excel
  • 原文地址:https://www.cnblogs.com/lindalee/p/9733349.html
Copyright © 2011-2022 走看看