当前位置:首页 > 力扣 > LeetCode 1031题解析:不重叠子数组最大和的解法(前缀和+动态规划)

LeetCode 1031题解析:不重叠子数组最大和的解法(前缀和+动态规划)

2个月前 (07-12)

LeetCode 1031题解析:不重叠子数组最大和的解法(前缀和+动态规划) 力扣题解 前缀和 动态规划 C++ 第1张

一、题目解读

LeetCode 1031题要求在不重叠的前提下,从给定数组nums中寻找两个长度分别为firSTLen和secondLen的连续子数组,使其和最大。题目强调子数组必须不重叠,即两个子数组的区间不能有交集,需要找到最优解。这一问题的关键在于高效计算所有可能的子数组组合并比较其和。

二、解题思路

采用前缀和+动态规划的解题思路。核心思想是利用前缀和数组简化子数组和的计算,再通过动态规划维护四个方向的最大子数组和,从而避免重复遍历。具体策略如下:

1. 计算前缀和数组prefixSum,方便O(1)时间获取任意区间和。

2. 定义四个方向的最大值数组:

○ firstMax:从左到右,长度为firstLen的子数组最大和;

○ secondMax:从左到右,长度为secondLen的子数组最大和;

○ firstMaxFromRight:从右到左,长度为firstLen的子数组最大和;

○ secondMaxFromRight:从右到左,长度为secondLen的子数组最大和。

3. 分别通过正向和逆向遍历计算上述四个数组,确保覆盖所有不重叠组合的可能情况。

4. 遍历所有合法位置,计算两个子数组和的组合最大值,并更新结果。

三、解题步骤

1. 计算前缀和:构建prefixSum数组,其中prefixSum[i+1]表示nums[0]到nums[i]的元素和。

2. 初始化动态规划数组:创建四个辅助数组,用于记录不同方向的最大子数组和。

3. 正向计算:从左到右遍历,更新firstMax和secondMax,利用当前子数组和与历史最大值比较更新结果。

4. 逆向计算:从右到左遍历,更新firstMaxFromRight和secondMaxFromRight,同样采用动态规划思路。

5. 遍历组合:通过两个指针分别指向两个子数组的起始位置,计算所有合法组合的和,取最大值作为最终结果。

四、代码及注释

class Solution {
public:
    int maxSumTwoNoOverlap(vector<int>& nums, int firstLen, int secondLen) {
        int n = nums.size();
        vector<int> prefixSum(n + 1, 0);
        
        // 计算前缀和数组
        for (int i = 0; i < n; ++i) {
            prefixSum[i + 1] = prefixSum[i] + nums[i];
        }
        
        // 初始化四个数组用于记录不同情况下的最大值
        vector<int> firstMax(n, 0);    // 记录到i位置为止firstLen长度的子数组最大值
        vector<int> secondMax(n, 0);   // 记录到i位置为止secondLen长度的子数组最大值
        vector<int> firstMaxFromRight(n, 0);  // 记录从i位置开始往右firstLen长度的子数组最大值
        vector<int> secondMaxFromRight(n, 0); // 记录从i位置开始往右secondLen长度的子数组最大值
        
        // 从左到右计算firstMax和secondMax
        for (int i = firstLen - 1; i < n; ++i) {
            int currentSum = prefixSum[i + 1] - prefixSum[i + 1 - firstLen];
            if (i == firstLen - 1) {
                firstMax[i] = currentSum;
            } else {
                firstMax[i] = max(firstMax[i - 1], currentSum);
            }
        }
        
        for (int i = secondLen - 1; i < n; ++i) {
            int currentSum = prefixSum[i + 1] - prefixSum[i + 1 - secondLen];
            if (i == secondLen - 1) {
                secondMax[i] = currentSum;
            } else {
                secondMax[i] = max(secondMax[i - 1], currentSum);
            }
        }
        
        // 从右到左计算firstMaxFromRight和secondMaxFromRight
        for (int i = n - firstLen; i >= 0; --i) {
            int currentSum = prefixSum[i + firstLen] - prefixSum[i];
            if (i == n - firstLen) {
                firstMaxFromRight[i] = currentSum;
            } else {
                firstMaxFromRight[i] = max(firstMaxFromRight[i + 1], currentSum);
            }
        }
        
        for (int i = n - secondLen; i >= 0; --i) {
            int currentSum = prefixSum[i + secondLen] - prefixSum[i];
            if (i == n - secondLen) {
                secondMaxFromRight[i] = currentSum;
            } else {
                secondMaxFromRight[i] = max(secondMaxFromRight[i + 1], currentSum);
            }
        }
        
        int result = 0;
        // 情况1:firstLen子数组在secondLen子数组左边
        for (int i = firstLen - 1; i < n - secondLen; ++i) {
            result = max(result, firstMax[i] + secondMaxFromRight[i + 1]);
        }
        
        // 情况2:secondLen子数组在firstLen子数组左边
        for (int i = secondLen - 1; i < n - firstLen; ++i) {
            result = max(result, secondMax[i] + firstMaxFromRight[i + 1]);
        }
        
        return result;
    }
};

五、总结

本解法通过前缀和数组大幅降低子数组和的计算复杂度,结合动态规划的思想维护不同方向的最大值,最终通过遍历所有合法组合找到最优解。时间复杂度O(n),空间复杂度O(n),高效解决了不重叠子数组求和问题。关键点在于利用前缀和的预处理与动态规划的双向计算,避免了暴力枚举的指数级时间消耗,是解决此类问题的经典思路。


原创内容 转载请注明出处

分享给朋友:

相关文章

力扣第71题:用栈轻松解决Unix路径简化问题

力扣第71题:用栈轻松解决Unix路径简化问题

题目解读:在Unix风格的文件系统中,我们经常需要处理各种复杂的路径表示。给定一个绝对路径字符串,我们需要将其转换为最简化的规范路径。规范路径要求:路径始终以斜杠'/'开头;两个目录名...

2024年GESP五级武器强化(洛谷B4071)解题代码C++版

2024年GESP五级武器强化(洛谷B4071)解题代码C++版

一、题目解读    2024年GESP(青少年软件编程能力等级考试)五级中的“武器强化”(洛谷平台题目编号B4071)是一道典型的算法优化问题。题目要求通过合理...

CSP-J 2019纪念品题解(洛谷P5662):动态规划+完全背包问题的实战应用

CSP-J 2019纪念品题解(洛谷P5662):动态规划+完全背包问题的实战应用

一、题目解读2019年CSP-J的“纪念品”问题(对应洛谷P5662)要求玩家在T天内通过买卖纪念品最大化金币收益。每天可交易N种商品,需计算最优策略下的最终金币数。题目强调动态规划思维与资源分配优化...

手搓顺序表类代码注释与详解:从零实现动态数组(新手教程)

一、简介和特点顺序表(Sequential List)是数据结构中基础的一种线性表,其特点是将数据元素存储在连续的内存空间中。通过数组实现,支持随机访问(即通过索引直接访问元素),适用于频繁随机读取的...

NOIP 2008火柴棒等式题解(C++代码实现)  动态规划与枚举算法详解

NOIP 2008火柴棒等式题解(C++代码实现) 动态规划与枚举算法详解

一、题目解读火柴棒等式问题(NOIP 2008,洛谷P1149)要求使用给定数量的火柴棒,构造形如 A + B = C 的等式,其中A、B、C均为整数,且火柴棒总数恰好等于输入值。需统计符合条件的等式...

2024蓝桥杯省赛B组“传送阵”题解(C++代码+图论算法优化)

2024蓝桥杯省赛B组“传送阵”题解(C++代码+图论算法优化)

一、题目解读2024年蓝桥杯省B组“传送阵”题目要求处理一个包含n个节点的图,节点间存在单向传输关系。每个节点i可传送至a[i]指定的节点,形成可能存在的环结构。题目需求解从任意节点出发能到达的最长路...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。