最优比率生成树 --- POJ --- 2728

题目链接

相关证明在另一篇简书上

题意就不说了,两个村庄之间的距离为欧几里得距离,花费是两个村庄的海拔高度.
求sigma(cost[i])/sigma(lenth[i]).
思路:
还是二分答案,使得答案不断接近我们要的那个答案.具体看代码吧.

#include<cstdio>
#include<cmath>
#include<algorithm>
#include<iostream>
#include<cstring>
#include<stack>
#include<cstdlib>
#define CLR(x) memset(x,0,sizeof(x))
#define ll long long int
#define PI acos(-1.0)
#define db double
#define mod 1000000007
using namespace std;
const int maxn=1e3+5;
const db eps=1e-7;
const db inf=1e9;
int n;
db mapp[maxn][maxn];
db low[maxn];
int vis[maxn];
struct point
{
    int x,y,z;
}s[maxn];

db cal(int m,int n)
{
    return sqrt(1.0 * (s[m].x-s[n].x)*(s[m].x-s[n].x)+1.0 * (s[m].y-s[n].y)*(s[m].y-s[n].y));
}

db prim(int fa,db mid)
{
    db sum=0;
    for(int i=1;i<=n;i++){
        low[i] = abs(s[fa].z-s[i].z)*1.0-mid*mapp[fa][i];   //d[i]数组.
    }
    CLR(vis);
    //cout << "BUG?" << endl;
    vis[fa]=1;
    for(int i=1;i<n;i++){  //找点肯定只用找n-1个点啊.
        db minn=inf;
        int v=-1;

        for(int j=1;j<=n;j++){
            if(!vis[j] && minn>low[j]){
                v=j;
                minn=low[j];
            }
            //cout << "BUG?" << " " << j << " " << n << endl;
        }

        if(v!=-1){
            sum += minn;
            vis[v]=1;
            for(int j=1;j<=n;j++){
                db tmp=abs(s[v].z-s[j].z)*1.0-mid*mapp[v][j];
                if(!vis[j] && low[j] > tmp)
                    low[j]= tmp;
            }
        }
    }
    //cout << "BUG?" << endl;
    return sum;
}

int main()
{
    while(~scanf("%d",&n) && n){
        for(int i=1;i<=n;i++){
            scanf("%d %d %d",&s[i].x,&s[i].y,&s[i].z);
        }
        CLR(mapp);
        for(int i=1;i<=n;i++){
            for(int j=1;j<=n;j++){
                mapp[i][j] =  cal(i,j);
            }
        }           // 不断测试,r=100是最好的.
        db l=0,r=100.0,mid;    //二分的范围自己定,总之越大当然肯定能出答案,但是时间相应也会增长,所以选择一个合理的范围是很重的.
        while(r-l>eps){
            mid = (r+l) / 2 ;
            if(prim(1,mid) >= 0 ) l=mid;
            else r=mid;
        }
        printf("%.3f\n",r);
    }

}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 背景 一年多以前我在知乎上答了有关LeetCode的问题, 分享了一些自己做题目的经验。 张土汪:刷leetcod...
    土汪阅读 12,774评论 0 33
  • 简介 搜索迷宫(BFS+队列) 最短路Dijkstra+邻接矩阵Dijkstra+链式前向星+优先队列Bellma...
    染微言阅读 431评论 0 1
  • G - Cyclic Tour题意:图中有n个点和m条有向边现在要将该图分成若干环,每个环中至少有两个点。环与环不...
    Gitfan阅读 814评论 0 1
  • 归去来兮。 1.1 说明 本篇为《挑战程序设计竞赛(第2版)》[http://www.ituring.com.cn...
    尤汐Yogy阅读 14,455评论 0 160
  • 1 我已经忘了,那天是什么样的天气,阴或晴。 脑海里满满的都是你要走的剧情,以至于忽略了周围存在的一切。 一路上我...
    安夏的花花世界阅读 249评论 0 1