leetcode刷题记录——Kadane算法

53 最大子数组和 Medium

给你一个整数数组 nums ,请你找出一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。

子数组

是数组中的一个连续部分。

示例 1:

1
2
3
输入:nums = [-2,1,-3,4,-1,2,1,-5,4]
输出:6
解释:连续子数组 [4,-1,2,1] 的和最大,为 6 。

示例 2:

1
2
输入:nums = [1]
输出:1

示例 3:

1
2
输入:nums = [5,4,-1,7,8]
输出:23

提示:

  • 1 <= nums.length <= 105
  • -104 <= nums[i] <= 104

思路

直接动态规划,记录每个位置的最大子数组和,进而推导出全局最大子数组和

状态转移方程$dp[i] = max(nums[i], nums[i] + dp[i - 1])$

代码

1
2
3
4
5
6
7
8
9
10
11
12
13
class Solution:
def maxSubArray(self, nums: List[int]) -> int:
n = len(nums)
if n == 0:
return 0
dp = [0] * n
dp[0] = nums[0]
for i in range(1, n):
dp[i] = max(nums[i], nums[i] + dp[i - 1])
res = float('-inf')
for i in range(n):
res = max(res, dp[i])
return res

918 环形子数组的最大和 Medium

给定一个长度为 n环形整数数组 nums ,返回 nums 的非空 子数组 的最大可能和

环形数组 意味着数组的末端将会与开头相连呈环状。形式上, nums[i] 的下一个元素是 nums[(i + 1) % n]nums[i] 的前一个元素是 nums[(i - 1 + n) % n]

子数组 最多只能包含固定缓冲区 nums 中的每个元素一次。形式上,对于子数组 nums[i], nums[i + 1], ..., nums[j] ,不存在 i <= k1, k2 <= j 其中 k1 % n == k2 % n

示例 1:

1
2
3
输入:nums = [1,-2,3,-2]
输出:3
解释:从子数组 [3] 得到最大和 3

示例 2:

1
2
3
输入:nums = [5,-3,5]
输出:10
解释:从子数组 [5,5] 得到最大和 5 + 5 = 10

示例 3:

1
2
3
输入:nums = [3,-2,2,-3]
输出:3
解释:从子数组 [3][3,-2,2] 都可以得到最大和 3

提示:

  • n == nums.length
  • 1 <= n <= 3 * 104
  • -3 * 104 <= nums[i] <= 3 * 104

思路

这题还是用Kadane算法去计算子数组的最大和,但需要注意几个问题。

首先我们在第一个的基础上额外用两个变量去存储当前位置的前一个位置的和(最大或最小)

其次我们还需要注意nums可能全为负

代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
class Solution:
def maxSubarraySumCircular(self, nums: List[int]) -> int:
n = len(nums)
if n == 0:
return 0

# 使用Kadane算法找到非环形子数组的最大和
max_sum = float('-inf')
current_max = 0
for num in nums:
current_max = max(num, current_max + num)
max_sum = max(max_sum, current_max)

# 找到非环形子数组的最小和
min_sum = float('inf')
current_min = 0
res = 0
for num in nums:
res += num
current_min = min(num, current_min + num)
min_sum = min(min_sum, current_min)

# 如果最大和大于0,则比较非环形最大和与环形最大和(总和减去最小和)
if max_sum > 0:
return max(max_sum, res - min_sum)
else:
# 如果所有数都是负数,则返回非环形最大和
return max_sum

时间复杂度

时间复杂度O(n),空间复杂度O(1),执行时间171ms,消耗内存20.4MB


leetcode刷题记录——Kadane算法
https://www.lx02918.ltd/2024/09/04/leetcode-Kadane算法/
作者
Seth
发布于
2024年9月4日
许可协议