title: 年终奖金 河南省第十届省赛
tags: [dp]

题目描述:

***公司承接了N个项目需要年底完成,每个项目有一定的难度系数。由于项目太多了,需要招聘大量的技术人员。要求每个技术人员至少完成K个项目。

考虑到有些项目之间相似性以及项目的难易程度,为了避免某些员工只挑选轻松项目,CEO提出了一个奖励机制,当技术人员完成分配给他的任务后,年终可以得到一笔奖金,其得到的酬金将是C + (Tmax–Tmin)2。其中,Tmax表示所做项目的最大的难度系数,Tmin是难度系数的最小值。

你能否计算一下,为了完成所有项目,***公司年终至少需要支付多少酬金?

输入:

输入有多组测试数据。对每组测试数据:

第一行: N  K  C     (1<=N,K<=100   1<=C<=5000 )

第二行   N个正整数分别描述N个项目的难度系数。(1<=难度系数<=10000)

输出:

对每组测试数据:输出占一行,一个整数。即,***公司年终至少需要支付的酬金数。

样例输入:

2 1 1
2 4
10 2 3
1 4 10 3 10 1 8 3 8 3

样例输出:

2
13

提示:

第一组测试数据,如果一个人完成,酬金为1 + (4–2)2 = 5;如果分给两个人去完成,收费为1 + 1 = 2。

分析:

首先肯定的是给出任务的难度是没有顺序的,首先肯定的给手游的任务按照难度从小到大排序,这样才能保证一个人完成的人物的难度差最小,然后还需要注意的一点就是一个人虽然最少要完成K个项目,但是却没有完成的个数上限,我们的任务只是计算如何分配才能保证给的工资最少,所以一个人最起码要完成K个项目,当前人与上一个人的任务号间隔最少也应该为K。

代码:

#include<stdio.h>
#include<string>
#include<string.h>
#include<algorithm>
#include<iostream>
using namespace std;
int main()
{
    int N,K,C,i,j;
    int num[105];
    int dp[105];
    while(~scanf("%d%d%d",&N,&K,&C))
    {
        memset(dp,0x3f3f3f3f,sizeof(dp));
        for(int i=1; i<=N; i++)
            scanf("%d",&num[i]);
        sort(num+1,num+N+1);
        dp[0]=0;
        for(i=K; i<=N; i++)///i从K才开始循环的原因就是,最起码要保证第一个人完成的项目有k个
        {
            for(j=0; j<=i-K; j++)///与前一个人之间最起码要有k个项目的间隔
            {
                if(j<K && j!=0)///j==0的时候不跳出循环的原因就是前面所有的项目都当作一个人完成的
                {
                    continue;
                }
                dp[i] = min(dp[j] + C + (num[i]-num[j+1])*(num[i]-num[j+1]),dp[i]);
            }
        }
        printf("%d\n",dp[N]);
    }
    return 0;
}
posted on 2017-05-22 07:40  渡……  阅读(238)  评论(0编辑  收藏  举报