当前位置: 首页 > backend >正文

动态规划题解——乘积最大子数组【LeetCode】

152. 乘积最大子数组

这道题的解法和思想和之前的一道题目非常相像,详情可见

数组题解——​最大子数组和​【LeetCode】(更新版)_leetcode 最大子数组-CSDN博客


一、算法逻辑(逐步思路)

❓ 题目描述:

给定一个整数数组 nums,找出一个连续子数组,使得乘积最大,返回这个最大乘积。


✅ 解题思路(动态规划)

1. 为什么需要同时维护最大值和最小值?
  • 因为数组中可能有 负数,当两个负数相乘时会变成正数;
  • 所以我们在每个位置都要维护两个状态:
    • f_max:以当前元素结尾的子数组的最大乘积;
    • f_min:以当前元素结尾的子数组的最小乘积。
2. 状态转移逻辑:

对当前元素 x,新的最大值和最小值可能由以下三者中的任一项决定:

  • f_max * x(当前数乘上之前的最大乘积);
  • f_min * x(当前数乘上之前的最小乘积 → 负数翻转变大);
  • x 本身(新起一个子数组);

所以转移写法为:

f_max, f_min = max(f_max * x, f_min * x, x), min(f_max * x, f_min * x, x)
3. 答案更新:

每次迭代后,用 ans = max(ans, f_max) 更新全局最大乘积。

4. 初始值设置:
  • ans = -inf:防止全是负数或只有一个负数的情况;
  • f_max = f_min = 1:初始乘积不影响第一位(或者你也可以初始化为 nums[0] 后从 nums[1:] 开始遍历)。

二、算法核心点

✅ 核心思想:乘积型动态规划 + 同时维护最大/最小子状态

  • 因为负数存在“翻转符号”的作用,所以不能只维护最大乘积;
  • 每一步都要考虑“以当前元素结尾”的最大/最小子数组;
  • 类似于“滑动窗口动态规划”,但只保留一对值来优化空间。
class Solution:def maxProduct(self, nums: List[int]) -> int:ans = -inff_max = f_min = 1for x in nums:f_max, f_min = max(f_max*x, f_min*x, x), min(f_max*x, f_min*x, x)ans = max(ans, f_max)return ans

三、复杂度分析

  • 时间复杂度:O(n)
    • 只遍历一遍数组,每次做常数次运算。
  • 空间复杂度:O(1)
    • 只使用常数个变量(f_maxf_minans),无额外空间开销。

总结表:

维度

内容

✅ 思路逻辑

遍历数组时维护以当前位置结尾的最大/最小乘积,更新全局最大值

✅ 核心技巧

负数会导致最大最小翻转,必须同时维护最大值和最小值

✅ 时间复杂度

O(n) 一次遍历

✅ 空间复杂度

O(1) 使用常数变量优化


✅ 举个例子帮助理解:

输入:nums = [2, 3, -2, 4]

i

x

f_max

f_min

ans

0

2

max(2,2,2)=2

min(2,2,2)=2

2

1

3

max(6,6,3)=6

min(3,3,3)=3

6

2

-2

max(-12,-6,-2)= -2

min(-12,-6,-2)=-12

6

3

4

max(-8,-48,4)=4

min(-8,-48,4)=-48

6

最终结果是 6,即子数组 [2,3] 的乘积。

http://www.xdnf.cn/news/15430.html

相关文章:

  • 暑期自学嵌入式——Day02(C语言阶段)
  • 【论文阅读】Masked Autoencoders Are Effective Tokenizers for Diffusion Models
  • 内容管理系统指南:企业内容运营的核心引擎
  • Kotlin Map映射转换
  • 论文阅读:WildGS-SLAM:Monocular Gaussian Splatting SLAM in Dynamic Environments
  • 院级医疗AI管理流程—基于数据共享、算法开发与工具链治理的系统化框架
  • ubuntu之坑(十八)——XML相关
  • CSS基础功能介绍和使用
  • Spring Boot项目结构解析:构建高效、清晰的代码框架
  • 关于僵尸进程
  • 进程、线程、协程
  • AI革命,分布式存储也在革命,全闪化拐点已至
  • MFC扩展库BCGControlBar Pro v36.2新版亮点:可视化设计器升级
  • 深入解析Paimon的RowKind数据变更机制 和 KeyValue存储
  • vue中使用西瓜播放器xgplayer (封装)+xgplayer-hls 播放.m3u8格式视频
  • 【王树森推荐系统】物品冷启05:流量调控
  • Java-72 深入浅出 RPC Dubbo 上手 生产者模块详解
  • 清除 Android 手机 SIM 卡数据的4 种简单方法
  • 网络准入控制系统的作用解析,2025年保障企业入网安全第一道防线
  • OpenVela之开发自测试框架cmocka
  • 【算法训练营Day12】二叉树part2
  • 量产技巧之RK3588 Android12默认移除导航栏状态栏​
  • google浏览器::-webkit-scrollbar-thumb设置容器滚动条滑块不生效
  • Android 性能优化:启动优化全解析
  • C++-linux 7.文件IO(一)系统调用
  • Linux上基于C/C++头文件查找对应的依赖开发库
  • uni-app 选择国家区号
  • CentOS 7服务器上使用Docker部署Notesnook的详细指导说明
  • Spring Cloud分布式配置中心:架构设计与技术实践
  • 链表算法之【获取链表开始入环的节点】