当前位置:首页 > 力扣 > 力扣2874题:动态规划求解最大乘积问题

力扣2874题:动态规划求解最大乘积问题

7个月前 (08-09)

力扣2874题:动态规划求解最大乘积问题 力扣题解 动态规划 C++ 第1张

一、题目解读

力扣2874题要求在一个整数数组中,找到三个不重叠子数组(可为空),使其乘积最大化。需确保子数组不相邻,且乘积结果为正数。当数组长度不足3时,返回0。题目核心在于如何高效划分数组,并计算合法组合的最大乘积。

二、解题思路

使用动态规划策略,解题分为三个关键步骤:

1. 预处理左右最大值数组:分别记录每个位置左侧/右侧的最大元素,确保子数组边界的最优选择;

2. 差值乘积计算:遍历中间位置j,利用左右最大值与当前值之差(left_max[j-1] - nums[j])与右侧最大值相乘,模拟三个子数组的乘积;

3. 动态更新最大值:遍历过程中实时记录最大合法乘积,并处理乘积为负的情况。

该思路通过空间换时间,避免暴力枚举的O(n^3)复杂度,实现线性时间求解。

三、解题步骤

1. 边界判断:当数组长度n<3时直接返回0,避免无效计算;

2. 预处理left_max数组:从前往后遍历,维护当前位置左侧的最大值(初始为nums[0]);

3. 预处理right_max数组:从后往前遍历,维护当前位置右侧的最大值(初始为nums[n-1]);

4. 核心遍历:

○ 对于中间位置j(1<j<n-1),计算(left_max[j-1] - nums[j]) * right_max[j+1],即左侧最大子数组与右侧最大子数组的差值乘积;

○ 动态更新全局最大值max_val,并确保结果为正(若最大值为负则保留0)。

四、代码与注释

class Solution {

public:

    long long maximumTripletValue(vector<int>& nums) {

        int n = nums.size();

        if (n < 3) return 0; // 边界处理:数组长度不足3时返回0

        

        // 预处理左边最大值数组

        vector<int> left_max(n);

        left_max[0] = nums[0];

        for (int i = 1; i < n; ++i) {

            left_max[i] = max(left_max[i-1], nums[i]); // 更新左侧最大值

        }

        

        // 预处理右边最大值数组

        vector<int> right_max(n);

        right_max[n-1] = nums[n-1];

        for (int i = n-2; i >= 0; --i) {

            right_max[i] = max(right_max[i+1], nums[i]); // 更新右侧最大值

        }

        

        long long max_val = 0;

        // 遍历j,计算差值乘积

        for (int j = 1; j < n-1; ++j) {

            long long current = (long long)(left_max[j-1] - nums[j]) * right_max[j+1];

            // 避免溢出:乘积可能超int范围,需强制转换

            max_val = max(max_val, current);

        }

        

        return max_val > 0? max_val : 0; // 若最大值负,返回0

    }

};

注释说明:

● 使用long long防止乘积溢出;

● 通过左右预处理数组,直接获取子数组边界的最优值;

● 最终结果需为正数,因此负值情况特判为0。

五、总结

通过动态规划将三维问题降为一维遍历,时间复杂度O(n),空间复杂度O(n)。核心在于利用预处理数组减少重复计算,确保每个位置只需一次访问即可确定最优解。该思路对类似“不重叠子数组问题”具有通用性,但需注意边界条件与数值溢出的处理。适合作为动态规划入门的典型案例,帮助理解“分治+预处理”的思想。


原创内容 转载请注明出处

分享给朋友:

相关文章

力扣451:ASCII数组计数法 用128个桶解决频率排序问题

力扣451:ASCII数组计数法 用128个桶解决频率排序问题

题目重解给定一个字符串,将字符按照出现频率降序排列。例如输入"tree",可能返回"eetr"或"eert"。题目要求我们不考虑字母顺序,只...

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

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

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

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]指定的节点,形成可能存在的环结构。题目需求解从任意节点出发能到达的最长路...

LeetCode 120题三角形最小路径和最优解法:动态规划详解与代码实现

LeetCode 120题三角形最小路径和最优解法:动态规划详解与代码实现

一、题目解读LeetCode 120题“三角形最小路径和”要求给定一个由数字组成的三角形,从顶部开始向下移动,每次可向左或向右移动一格,计算从顶至底的最小路径和。三角形以二维向量形式给出,每层元素数量...

CSP-J 2019公交换乘题解析:基于队列优化的动态规划代码详解

CSP-J 2019公交换乘题解析:基于队列优化的动态规划代码详解

一、题目解读CSP-J 2019年的“公交换乘”题目(洛谷P5661)要求模拟地铁与公交交替出行的费用计算。题目核心在于地铁消费会产生优惠券,而公交可在45分钟内使用优惠券抵扣车费。需要处理n条出行记...

发表评论

访客

看不清,换一张

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