UVA 10163 - Storage Keepers(dp)

题意:

有n个仓库(最多100个),m个管理员(最多30个),
每个管理员有一个能力值P(接下来的一行有m个数,表示每个管理员的能力值),
每个仓库只能由一个管理员看管,但是每个管理员可以看管k个仓库(但是这个仓库分配到的安全值只有p/k,k=0,1,…),每个月公司都要给看管员工资,雇用的管理员的工资即为他们的能力值p和。
问:使每个仓库的安全值最高的前提下,使的工资总和最小。
输出最大安全值,并且输出最少的花费。

解析:

dp[i][j]表示前i个人,管理j个仓库的最大安全值。
dp[i][j] = max{ min{dp[i-1][j-k], p[i]/k}, 0<=k<=j && k是第i个人管理的仓库个数 }
然后求最少价钱,
f[i][j]表示前i个人,管理j个仓库的最大安全值下所用的最少价钱
f[i][j] = min{ f[i-1][j-k]+p[i], p[i]/k>=dp[m][n] && 0<=k<=j }

AC代码:

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <cstdlib>
using namespace std;
typedef long long ll;
const int INF = 0x3f3f3f3f;
const int N = 105;
const int M = 35;
int dp[M][N] ,f[M][N] ,p[M];
int n, m;
void init() {
    memset(dp,0,sizeof(dp));
    memset(f,INF,sizeof(f));
    for(int i = 0; i <= m; i++) {
        dp[i][0] = INF;
        f[i][0] = 0;
    }
}
int main() {
    while(scanf("%d%d",&n, &m) != EOF && (m || n)) {
        init(); 
        for(int i = 1; i <= m; i++) {
            scanf("%d",&p[i]);
        }
        for(int i = 1; i <= m; i++) {
            for(int j = 1; j <= n; j++) {
                dp[i][j] = dp[i-1][j];
                for(int k = 1; k <= j; k++) {
                    dp[i][j] = max(dp[i][j], min(dp[i-1][j-k], p[i] / k));
                }
            }
        }
        for(int i = 1; i <= m; i++) {
            for(int j = 1; j <= n; j++) {
                f[i][j] = f[i-1][j];
                for(int k = 1; k <= j; k++) {
                    if(p[i] / k >= dp[m][n]) {
                        f[i][j] = min(f[i][j], f[i-1][j-k] + p[i]);
                    }
                }
            }
        }
        printf("%d %d\n", dp[m][n], (dp[m][n] ? f[m][n] : 0) );
    }
    return 0;
}

你可能感兴趣的:(uva,10163)