leetcode813.最大平均值和的分组

题目大意

我们将给定的数组 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();
    }
};

你可能感兴趣的:(leetcode算法题解)