SRM 502 DIV1 500pt(DP)

题目简述node

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

题解spa

因为问题解决的前后,得到的分数是不同的,由于咱们首先得肯定作题的选择顺序,根据相邻交换法,对于问题a和问题b,若是requiredTime[b]*pointsPerMinute[a]>requiredTime[a]*pointsPerMinute[b],先解决a再解决b得到的分数会比先b后a多,因此咱们能够先根据这个条件对题目进行排序,以后就是01背包问题了code

代码:blog

 1 struct node
 2 {
 3     int mp, pp, rt;
 4 };
 5 node a[55];
 6 int  dp[100005];
 7 bool cmp(node a, node b)
 8 {
 9     return (LL)b.rt * a.pp > (LL)a.rt * b.pp;
10 }
11 class TheProgrammingContestDivOne
12 {
13 public:
14     LL find(int T, vector <int> maxPoints, vector <int> pointsPerMinute, vector <int> requiredTime)
15     {
16         int n = maxPoints.size();
17         for (int i = 0; i < n; i++)
18         {
19             a[i + 1].mp = maxPoints[i];
20             a[i + 1].pp = pointsPerMinute[i];
21             a[i + 1].rt = requiredTime[i];
22         }
23         sort(a + 1, a + n + 1, cmp);
24         memset(dp, 0, sizeof(dp));
25         for (int i = 1; i <= n; i++)
26             for (int j = T; j >= a[i].rt; j--)
27                 if (a[i].mp - (LL)a[i].pp * j >=0)
28                     dp[j] = max(dp[j], dp[j - a[i].rt] + a[i].mp - a[i].pp * j);
29         int ans = 0;
30         for (int i = 0; i <= T; i++) ans = max(ans, dp[i]);
31         return ans;
32     }
33 };
相关文章
相关标签/搜索