LeetCode152. 乘积最大子数组(2024秋季每日一题 2)

给你一个整数数组 n u m s nums nums,请你找出数组中乘积最大的非空连续 子数组(该子数组中至少包含一个数字),并返回该子数组所对应的乘积。

测试用例的答案是一个 32 32 32-位 整数。

示例 1:

复制代码
输入: nums = [2,3,-2,4]
输出: 6
解释: 子数组 [2,3] 有最大乘积 6。

示例 2:

复制代码
输入: nums = [-2,0,-1]
输出: 0
解释: 结果不能为 2, 因为 [-2,-1] 不是子数组。

提示:

1 < = n u m s . l e n g t h < = 2 ∗ 1 0 4 1 <= nums.length <= 2 * 10^4 1<=nums.length<=2∗104
− 10 < = n u m s i < = 10 -10 <= numsi <= 10 −10<=numsi<=10
n u m s nums nums 的任何子数组的乘积都 保证 是一个 32 32 32-位 整数


方法一:暴力: O ( N 2 ) O(N^2) O(N2)

能过的样例数: 187 / 190 187/190 187/190

思路:第一层循环枚举子数组长度,第二层循环枚举子数组区间左边的索引,并且维护一个滑动窗口,用于记录窗口内的乘积值。

cpp 复制代码
class Solution {
public:
    int maxProduct(vector<int>& nums) {
        int res = -20;
        for(int len = 1; len <= nums.size(); len++){
            int x = 1, l = -1;
            for(int j = 0; j <= len - 1; j++){
                if(x == 0) x = nums[j];
                else x *= nums[j];
                if(nums[j] == 0) l = j;
            }
            res = max(res, x);
            for(int i = 1; i + len - 1 < nums.size(); i++){
                if(x == 0) x = nums[i + len - 1];
                else {
                    if(nums[i + len - 1] == 0) l = i + len - 1;
                    if(nums[i-1] != 0 && i > l){
                        x = x / nums[i-1] * nums[i + len - 1];
                    }else{
                        x = x * nums[i + len - 1];
                    }
                }
                res = max(res, x);
            }
        }
        return res;
    }
};

方法二:DP: O ( N ) O(N) O(N)

思路:维护三个数,res、imax、imin

  • 遍历数组时维护当前子数组的最大值、最小值
  • 当遇到负数时,最大值,最小值交换
  • 当遇到0时,imax,imin 重置,在子数组内重新计算

注意:这里的子数组指的是 以 0 为分割的子数组

i m a x = m a x ( n u m s i , i m a x ∗ n u m s i ) imax = max(numsi, imax * numsi) imax=max(numsi,imax∗numsi)
i m i n = m i n ( n u m s i , i m i n ∗ n u m s i ) imin = min(numsi, imin * numsi) imin=min(numsi,imin∗numsi)

当乘给 负数 时,imin 与 imax 交换(最大,最小值交换)

cpp 复制代码
class Solution {
public:
    int maxProduct(vector<int>& nums) {
        int res = -20, imin = 1, imax = 1;
        for(int i = 0; i < nums.size(); i++){
            if(nums[i] < 0) swap(imin, imax);
            imax = max(imax * nums[i], nums[i]);
            imin = min(imin * nums[i], nums[i]);

            res = max(res, imax);
        }
        return res;
    }
};
相关推荐
咸鱼翻身小阿橙6 小时前
高斯模糊降噪/磨皮算法降噪图像
前端·opencv·算法·webpack·c#
代码中介商6 小时前
数据结构进阶(五):最短路径——Dijkstra 与 Floyd 算法
数据结构·算法
fengxin_rou6 小时前
LeetCode链表经典五题:从相交到环形,双指针的妙用
算法·leetcode·链表
KaMeidebaby6 小时前
卡梅德生物技术快报|抗原如何自己检测?FAdV-4 重组抗原制备与 ELISA 体系技术调试指南
前端·人工智能·物联网·算法·百度
却道天凉_好个秋6 小时前
HEVC(二):如何实现并行处理
人工智能·算法·计算机视觉·hevc·瓦片技术·波前并行处理wpp
wayz116 小时前
Momentum:QQE(定量定性估计)技术指标详解
算法·金融·数据分析·量化交易·特征工程
Dontla6 小时前
聚类找不到簇原因分析(聚类失败)(DBSCAN聚类算法、eps参数、Epsilon参数、最大允许距离)
算法·数据挖掘·聚类
写代码写到手抽筋14 小时前
5G上行DCI字段判定:端口 流数 PMI选择详解
java·算法·5g
xieliyu.14 小时前
Java算法精讲:双指针(二)
java·开发语言·算法
wayz1115 小时前
Momentum:PSL(心理线指标)技术指标详解
算法·金融·数据分析·量化交易·特征工程