【Uva 10163】Storage Keepers

Link:

Description

你有n(n≤100)个相同的仓库。有m(m≤30)个人应聘守卫,第i个应聘者的能力值 为Pi(1≤Pi≤1000)。每个仓库只能有一个守卫,但一个守卫可以看守多个仓库。如果应聘 者i看守k个仓库,则每个仓库的安全系数为Pi/K的整数部分。没人看守的仓库安全系数为0。
你的任务是招聘一些守卫,使得所有仓库的最小安全系数最大,在此前提下守卫的能力 值总和(这个值等于你所需支付的工资总和)应最小。

Solution

动规
能力总和最大值为30000
安全系数[0..1000]
100个仓库,30个应聘人
设f[i][k][j]表示
前i个应聘人,安排好了k个仓库,最小安全系数为j最小能力总和
枚举第i个人选择了多少个仓库;
做一下状态转移就好;
(顺推比较好写吧)
如果没有安排满n个仓库,直接输出”0 0”就好

NumberOf WA

0

Reviw

找题目中的一些参数,尝试用那些参数作为动态规划的状态;

Code

#include <bits/stdc++.h>
#define int long long
using namespace std;

const int M = 30;
const int K = 100;
const int S = 1000;
const int INF = 0x3f3f3f3f;

int f[M+5][K+5][S+5],p[M+5],n,m;

main(){
    //freopen("F:\\rush.txt","r",stdin);
    while (~scanf("%lld%lld",&n,&m)){
        if (n==0 && m==0) break;
        for (int i = 1;i <= m;i++)
            scanf("%lld",&p[i]);
        memset(f,INF,sizeof f);
        f[0][0][1001] = 0;
        for (int i = 0;i <= m-1;i++)
            for (int j = 0;j <= n;j++)//枚举已经安排好的仓库个数
                for (int k = 0;k <= 1001;k++) //枚举最小安全值
                    if (f[i][j][k] < INF){
                        for (int tj = 0;tj <= n-j;tj++){//枚举第i+1个人要守卫几个
                            //i+1 j+tj min(k,p[i+1]/tj)
                            if (tj==0){
                                f[i+1][j][k] = min(f[i+1][j][k],f[i][j][k]);
                            }else{
                                f[i+1][j+tj][min(k,p[i+1]/tj)] = min(
                                    f[i+1][j+tj][min(k,p[i+1]/tj)],f[i][j][k]+p[i+1]);
                            }
                        }
                    }
        int ans = INF,pos = 1001;
        for (int i = 1000;i >= 1;i--)
            if (f[m][n][i]<INF){
                ans = f[m][n][i];
                pos = i;
                break;
            }
        if (ans<INF){
            printf("%lld %lld\n",pos,ans);
        }else{
            puts("0 0");
        }
    }
    return 0;
}

转载于:https://www.cnblogs.com/AWCXV/p/7626161.html

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值