SRM 502 DIV1 500pt(DP)

题目简述

给定比赛时间T和n个题目,你可以在任意时间提交题目,每个题目有一个初始分数maxPoints[i],每个单位时间题目的分数将会减少pointsPerMinute[i],即如果在时间t解决了第i个题目,那么获得的分数为maxPoints[i] - t * pointsPerMinute[i],另外做每个题目需要requiredTime[i]的时间,求能够获得的最大分数是多少?

题解

由于问题解决的先后,获得的分数是不一样的,因为我们首先得确定做题的选择顺序,根据相邻交换法,对于问题a和问题b,如果requiredTime[b]*pointsPerMinute[a]>requiredTime[a]*pointsPerMinute[b],先解决a再解决b获得的分数会比先b后a多,所以我们可以先根据这个条件对题目进行排序,之后就是01背包问题了

代码:

 struct node
{
int mp, pp, rt;
};
node a[];
int dp[];
bool cmp(node a, node b)
{
return (LL)b.rt * a.pp > (LL)a.rt * b.pp;
}
class TheProgrammingContestDivOne
{
public:
LL find(int T, vector <int> maxPoints, vector <int> pointsPerMinute, vector <int> requiredTime)
{
int n = maxPoints.size();
for (int i = ; i < n; i++)
{
a[i + ].mp = maxPoints[i];
a[i + ].pp = pointsPerMinute[i];
a[i + ].rt = requiredTime[i];
}
sort(a + , a + n + , cmp);
memset(dp, , sizeof(dp));
for (int i = ; i <= n; i++)
for (int j = T; j >= a[i].rt; j--)
if (a[i].mp - (LL)a[i].pp * j >=)
dp[j] = max(dp[j], dp[j - a[i].rt] + a[i].mp - a[i].pp * j);
int ans = ;
for (int i = ; i <= T; i++) ans = max(ans, dp[i]);
return ans;
}
};
上一篇:JS---计算两个日期之间相差多少天


下一篇:Java 计算两个日期相差的天数