题目大意
我们将给定的数组 A 分成 K 个相邻的非空子数组 ,我们的分数由每个子数组内的平均值的总和构成。计算我们所能得到的最大分数是多少。
注意我们必须使用 A 数组中的每一个数进行分组,并且分数不一定需要是整数。
示例:
输入:
A = [9,1,2,3,9]
K = 3
输出: 20
解释:
A 的最优分组是[9], [1, 2, 3], [9]. 得到的分数是 9 + (1 + 2 + 3) / 3 + 9 = 20.
我们也可以把 A 分成[9, 1], [2], [3, 9].
这样的分组得到的分数为 5 + 2 + 6 = 13, 但不是最大值.
说明:
- 1 <= A.length <= 100.
- 1 <= A[i] <= 10000.
- 1 <= K <= A.length.
- 答案误差在 10-6 内被视为是正确的。
解题思路
动态规划。计算前i个元素划分为k个子数组的最大和时,只需要计算前j个元素(j<=i-1)划分为k-1个子数组的最大和,然后求得j+1~i范围内元素的均值相加即可。dp[i][k]可表示前i个元素划分为k个子数组时的最大平均值和。
class Solution {
public:
double largestSumOfAverages(vector<int>& A, int K) {
int length = A.size();
vector<double> accu(length, 0);
accu[0] = A[0];
// 计算前i项的累加和
for (int i = 1; i < length; ++i)
accu[i] = accu[i - 1] + A[i];
// dp[i][k]表示前i个元素,划分为k+1个子数组(因为k的索引从0开始),最大的平均值的和
vector<vector<double>> dp(length, vector<double>(K, 0));
for (int i = 0; i < length; ++i){
// 只划分为1个子数组时,前i项的累加和的平均值就是结果
dp[i][0] = accu[i] / (i + 1);
// 当划分为k+1个子数组时
for (int k = 1; k < K && k <= i; ++k){
// 前j个元素划分为k个子数组的最大和,加上j~i元素累加和的平均值,找最大
// 因为要划分为k个子数组,所以j不能小于k-1
for (int j = i - 1; j >= k - 1; --j){
dp[i][k] = max(dp[i][k], dp[j][k - 1] + (accu[i] - accu[j]) / (i - j));
}
}
}
return dp.back().back();
}
};