01分数规划 观光奶牛

01分数规划的题目通常长这样

image-20220227151949154

本质上是在求 ∑ f i ∑ t i \frac{\sum{f_i}}{\sum{t_i}} tifi的最大值,这类分数求最大、最小的问题被成为01分数规划问题。

01分数规划求最大值通常使用二分,二分的范围要看题目,对于本题 ∑ 环 点 权 ∑ 环 边 权 \frac{\sum{环点权}}{\sum{环边权}} 的范围为 ( 0 , 1000 ] (0, 1000] (0,1000]

于是问题可以转化为问号处是否成立, ∑ f i ∑ t i    > ?    m i d    ⇔ ∑ ( f i    −    m i d ⋅ t i )    > 0    \frac{\sum{f_i}}{\sum{t_i}}\,\,\overset{?}{>}\,\,mid\,\,\Leftrightarrow \sum{\left( f_i\,\,-\,\,mid\cdot t_i \right)}\,\,>0\,\, tifi>?mid(fimidti)>0 于是就是将所有边权变为点权 - mid * 边权,求是否有正环,而求环的存在性问题可以用spfa,这样问题就解决了

/*
 * @Author: 爱学习的图灵机
 * @Date: 2022-02-20 16:48:10
 * @LastEditTime: 2022-02-20 20:43:01
 * Bilibili:https://space.bilibili.com/7469540
 * 题目地址:https://www.acwing.com/problem/content/363/
 * @keywords:  spfa 01分数规划
 */
#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;
const int N = 1010,M = 5010;
int h[N], e[M], wt[M], ne[M], idx;
int q[N];
double dist[N];
bool st[N];
int wf[N];// 点权
int cnt[N]; 

int n, m;

void add(int a,int b,int c){
    e[idx] = b, wt[idx] = c, ne[idx] = h[a],h[a] = idx ++;
}

bool check(double mid){ //check要执行多次,要初始化
//!double mid
    memset(dist, 0, sizeof dist);// 可以不要
    memset(st, 0, sizeof st);
    memset(cnt, 0, sizeof cnt);
    
    int hh = 0, tt = 0;
    for(int i = 1; i <= n; ++ i){
        st[i] = true;
        q[tt ++] = i;
    }
    
    while(hh != tt){
        int t = q[hh ++]; if(hh == N) hh = 0;
        st[t] = false;
        
        for(int i = h[t]; ~ i; i = ne[i]){
            int j = e[i];
            if(dist[j] < dist[t] + wf[t] - mid * wt[i]){
                dist[j] = dist[t] + wf[t] - mid * wt[i];
                cnt[j] = cnt[t] + 1;
                if(cnt[j] >= n)return true;
                
                if(!st[j]){
                    q[tt ++] = j;
                    if(tt == N) tt = 0;
                    st[j] = true;
                }
    
            }
        }
    }
    return false;
}

int main()
{
    memset(h, -1, sizeof h);
    cin >> n >> m;
    for(int i = 1; i <= n; ++ i){
        cin >> wf[i];
    }
    
    for(int i = 1; i <= m; ++ i){
        int a, b, c;
        cin >> a >> b >> c;
        add(a, b, c);
    }
    
    double l = 0,r = 1010;
    while(r - l > 1e-4){
        double mid = (l + r) / 2;
        
        if(check(mid)){
            l = mid;
        }else {
            r = mid;
        }
    }
    printf("%.2lf", r);
    
    return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值