赞
踩
给你一个整数数组 nums ,请你找出数组中乘积最大的非空连续
子数组
(该子数组中至少包含一个数字),并返回该子数组所对应的乘积。
测试用例的答案是一个 32-位 整数。
示例 1:
输入: nums = [2,3,-2,4]
输出: 6
解释: 子数组 [2,3] 有最大乘积 6。
示例 2:
输入: nums = [-2,0,-1]
输出: 0
解释: 结果不能为 2, 因为 [-2,-1] 不是子数组。
提示:
1 <= nums.length <= 2 * 104
-10 <= nums[i] <= 10
nums 的任何前缀或后缀的乘积都 保证 是一个 32-位 整数
遇到这样的问题,其实就在提示我们状态不够用了。因此,需要在原来的一维 dp 后面新增一个状态。
针对这道题,第 2 维状态只需要两个:
当 nums[i] = 0 的时候包含在上面二者之中,无需单独讨论。
状态转移方程写在了参考代码 1 中。即使用二维状态数组同时记录乘积的最大值和最小值,本来写了一堆文字的,后来看太长了,好多废话,直接看代码比较清楚一些。
这里就声明一下状态:
dp[i][1] 表示:以 nums[i] 结尾的连续子序列的乘积的最大值;
dp[i][0] 表示:以 nums[i] 结尾的连续子序列的乘积的最小值。
if nums[i] >= 0:
dp[i][1] = max(nums[i], dp[i-1][1] * nums[i])
dp[i][0] = min(nums[i], dp[i-1][0] * nums[i])
else:
dp[i][1] = max(nums[i], dp[i-1][0] * nums[i])
dp[i][0] = max(nums[i], dp[i-1][1] * nums[i])
dp = [[0, 0] for _ in range(n)]
dp[0][0] = nums[0]
dp[0][1] = nums[0]
for i in range(1, n):
class Solution: def maxProduct(self, nums: List[int]) -> int: n = len(nums) if n == 0: return 0 dp = [[0, 0] for _ in range(n)] dp[0][0] = nums[0] dp[0][1] = nums[0] for i in range(1, n): if nums[i] >= 0: dp[i][1] = max(nums[i], dp[i-1][1] * nums[i]) dp[i][0] = min(nums[i], dp[i-1][0] * nums[i]) else: dp[i][1] = max(nums[i], dp[i-1][0] * nums[i]) dp[i][0] = min(nums[i], dp[i-1][1] * nums[i]) # print(dp) ans = dp[0][1] for i in range(1, n): ans = max(ans, dp[i][1]) return ans
时间复杂度:O(n)
空间复杂度:O(n)
class Solution: def maxProduct(self, nums: List[int]) -> int: n = len(nums) if n == 0: return 0 # dp = [[0, 0] for _ in range(n)] # dp[0][0] = nums[0] # dp[0][1] = nums[0] preMax = nums[0] preMin = nums[0] curMax = preMax curMin = preMin ans = nums[0] for i in range(1, n): if nums[i] >= 0: curMax = max(nums[i], preMax * nums[i]) curMin = min(nums[i], preMin * nums[i]) else: curMax = max(nums[i], preMin * nums[i]) curMin = min(nums[i], preMax * nums[i]) ans = max(ans, curMax) # 滚动变量 preMax = curMax preMin = curMin return ans
时间复杂度:O(n)
空间复杂度:O(1)
时间复杂度:O(n)
空间复杂度:O(n)
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。