Desert King POJ2728(Prim+迭代+0-1分数规划)

【原题地址】: POJ2728

【题目大意】:

\(n-1\) 条边,最小化这些边的\[\frac{\sum w_i} {\sum d_i}\]

其中 \(w_i\) 为第 \(i\) 条边的花费,\(d_i\) 为这条边所连接的两个点的距离。
\(w_i\) 为连接的两个点的 \(z\) 值差的绝对值, \(d_i\) 为欧几里得距离

【题解】:

这道题是0-1分数规划的经典题目
我们令\[\frac{\sum w_i} {\sum d_i}=k\]
这样二分 \(k\) 的值
将所有的边权改为 \(w_i - k*d_i\)
Prim 求出最小生成树
注: 这道题负边很多,用堆优化prim会超时
若最小生成树边权总和 \(≥ 0\),则说明 \(k\) 值小,否则 \(k\) 值偏大。
如果用二分可能会超时
我们设 \(MST(now)\)\(k=now\) 时最小生成树的结果
那么可以发现当 \(now\) 越大则 \(MST(now)\) 值越小
所以我们采用迭代,用上次计算的值,计算当前值

【AC代码】:

#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
using namespace std;
const int maxl=1e3+10;
int n;
int abs(int x)
{
    return x>0?x:-x;
}
double dis[maxl][maxl];
int x[maxl],y[maxl],z[maxl],cost[maxl][maxl];
int vis[maxl];
double to[maxl];
double tot_cost,tot_dis;
int reach[maxl];
void prim(double mid)
{
    vis[1]=1;
    for(int i=2;i<=n;i++)to[i]=cost[1][i]-dis[1][i]*mid,vis[i]=0,reach[i]=1;
    tot_cost=tot_dis=0;
    for(int i=1;i<=n-1;i++)
    {
        int pos;
        double minn=1e50;
        for(int j=2;j<=n;j++)
        if(!vis[j])
        {
            if(minn>to[j])
            {
                minn=to[j];
                pos=j;
            }
        }
        vis[pos]=1;
        tot_cost+=cost[reach[pos]][pos];
        tot_dis+=dis[reach[pos]][pos];
        for(int j=2;j<=n;j++)
        if(!vis[j]){
            if(to[j]>cost[pos][j]-dis[pos][j]*mid)to[j]=cost[pos][j]-dis[pos][j]*mid,reach[j]=pos;
        }
    }
}
int main()
{
    while(~scanf("%d",&n)&&n)
    {
        for(int i=1;i<=n;i++)
        scanf("%d%d%d",&x[i],&y[i],&z[i]);
        for(int i=1;i<=n;i++)
        for(int j=i;j<=n;j++)
        dis[i][j]=dis[j][i]=sqrt((x[i]*1.0-x[j])*(x[i]*1.0-x[j])+(y[i]*1.0-y[j])*(y[i]*1.0-y[j])),cost[i][j]=cost[j][i]=abs(z[i]-z[j]);
        double ans=0,last=1e100;
        while(abs(ans-last)>1e-4)
        {
            prim(ans);
            last=ans;
            ans=tot_cost/tot_dis;
        }
        printf("%.3f\n",ans);
        
    }
}

转载于:https://www.cnblogs.com/Harry-bh/p/8798435.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值