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

【Leetcode152】乘积最大子数组(动态规划)

文章目录

  • 一、题目
  • 二、思路
  • 三、代码

一、题目

在这里插入图片描述

二、思路

(0)读懂题意:题目的“连续”是指位置的连续,而不是说数字的连续,这是个大坑。

(1)确定状态:定义两个状态来记录当前子数组的最大乘积、最小乘积。因为在处理负数时,最小乘积乘以负数可能变为最大乘积。dp_max[i]表示以nums[i]结尾的子数组的最大乘积、dp_min[i]表示以nums[i]结尾的子数组的最小乘积。

(2)状态转移方程:对于每个元素nums[i],我们的dp_max[i]dp_min[i]可以从这三个数中确定:

  • 只包含当前元素 nums[i]
  • 当前元素与之前的最大乘积子数组乘积,即 dp_max[i-1] * nums[i]
  • 当前元素与之前的最小乘积子数组乘积,即 dp_min[i-1] * nums[i]

即状态转移方程可表示为:

dp_max[i] = max(nums[i], dp_max[i-1] * nums[i], dp_min[i-1] * nums[i])
dp_min[i] = min(nums[i], dp_max[i-1] * nums[i], dp_min[i-1] * nums[i])

(3)初始状态+边界条件:以第一个元素结尾的子数组最大乘积就是它本身、以第一个元素结尾的子数组最小乘积就是它本身、初始乘积最大结果为第一个元素。

(4)遍历顺序:从左到右遍历数组。

三、代码

(方法一)按照思路的代码如下,时间复杂度为O(n),空间复杂度为O(n)。

class Solution(object):
    def maxProduct(self, nums):
        """
        :type nums: List[int]
        :rtype: int
        """
        if len(nums) == 0:
            return 0
        if len(nums) == 1:
            return nums[0]
        # 初始化数组
        n = len(nums)
        dp_max = [0] * n 
        dp_min = [0] * n
        # 初始状态
        dp_max[0] = nums[0]
        dp_min[0] = nums[0]
        cheng_ans = nums[0]
        # 从第二个元素开始遍历
        for i in range(1, n):
            num = nums[i]
            dp_max[i] = max(num, dp_max[i-1]*num, dp_min[i-1]*num)
            dp_min[i] = min(num, dp_max[i-1]*num, dp_min[i-1]*num)
            cheng_ans = max(cheng_ans, dp_max[i])
        return cheng_ans

(方法二)为了优化空间复杂度,发现每次当前只利用前一次状态,所以dp_maxdp_min没必要单独用两个数组记录所有的状态。但注意在计算状态转移方程时,分别计算dp_max和dp_min都会用到上一次的dp_max和dp_min,这为了用错dp_mxn,可以直接对num确保是正数后,交换dp_max和dp_min的位置,减少max和min函数的入参个数。

class Solution(object):
    def maxProduct(self, nums):
        """
        :type nums: List[int]
        :rtype: int
        """
        if len(nums) == 0:
            return 0
        if len(nums) == 1:
            return nums[0]
        # 初始化数组
        n = len(nums)
        # 初始状态
        dp_max = nums[0]
        dp_min = nums[0]
        cheng_ans = nums[0]
        # 从第二个元素开始遍历
        for i in range(1, n):
            num = nums[i]
            if num < 0:
                dp_max, dp_min = dp_min, dp_max
            dp_max = max(num, dp_max*num)
            dp_min = min(num, dp_min*num)
            cheng_ans = max(cheng_ans, dp_max)
        return cheng_ans

http://www.kler.cn/a/301372.html

相关文章:

  • 黑马智数Day10
  • 初识Linux · 信号处理 · 续
  • PyTorch数据集方法
  • 【linux】如何扩展磁盘容量(VMware虚拟机)-转载
  • [ 网络安全介绍 3 ] 网络安全事件相关案例有哪些?
  • Excel根据条件动态索引单元格范围
  • 音乐项目
  • JVM源码解析
  • 20道经典自动化测试面试题【建议收藏】
  • SpringMVC重点功能底层源码解析
  • Rocky Linux 9 初次安装后启用 SSH Root 远程登录
  • 使用Docker快速启动MySQL容器
  • Python知识点:如何使用Python进行文件压缩与解压缩
  • 证书学习(四)X.509数字证书整理
  • springcloud-GateWay
  • 大语言模型:AI 时代的文字计算器?
  • Vue2和Vue3子组件向父组件传值
  • INIC6081量产工具下载,initio6081开卡软件分享
  • 前端自定义下载文件名
  • 第一篇 第3章 不确定型分析 第4章 设备更新分析 第5章价值工程
  • 基于SpringBoot+Vue的超市外卖管理系统
  • vue3使用provide和inject传递异步请求数据子组件接收不到
  • 大文件切片上传-vue3.0
  • 如何在VUE3中使用函数式组件
  • ecmascript和javascript的区别?
  • 从底层原理上理解ClickHouse 中的稀疏索引