【刷题日记】最大子数组问题
在学习算法时,很多书籍资料都是按照类别进行分类学习的,比如先学分治、动态规划,再学贪心等。本博客的部分文章将根据作者本人的刷题经历,以典型题、模板题、系列题进行总结与发散,站在另一个角度审视这些题目,从而看清问题的本质与事物的全貌。
经典黑书《算法导论》在介绍分治算法时选择了以最大子数组问题为教学案例进行讲解,本文将继续通过这个问题的解决方案进行发散探讨,以串联各种各样的知识。
最大子数组问题:给定数组 A,寻找其中一个 「和最大的子数组」。只有数组中包含负数时,最大子数组问题才有意义。
本文题目难度标识:🟩简单,🟨中等,🟥困难。
给你一个整数数组 nums
,请你找出一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。
子数组:数组中的一个连续非空序列。
解决方法
传统分治解法
假定我们要寻找子数组 的最大子数组,利用分治技术划分为两个规模尽量相等的子数组 。 的一个最大子数组 必然是以下三种情况之一:
- 完全位于子数组 (左子数组)中,因此
- 完全位于子数组 (右子树组)中,因此
- 跨越了中点,因此
前两种情况实际上仍是最大数组问题,只是规模更小。我们的工作就是寻找跨越中点的最大子数组,然后在三种情况中选取和最大者。
我们可以在线性时间内求出跨越中点的最大子数组:
FIND-MAX-CROSSING-SUBARRAY
花费时间 。
求解最大子数组问题的分治算法的伪代码:
1 | FIND-MAXMUM-SUBARRAY(A,low,high) |
运行时间的递归式:
解为 。
空间复杂度:,需要常数个变量用于选取最大值,需要使用的空间取决于递归栈的深度。
动态规划
假设 nums
数组的长度是 n
,下标从 0
到 n-1
。
我们用 代表以第 i 个数结尾的「连续子数组的最大和」,那么很显然我们要求的答案就是:
因此我们只需要求出每个位置的 ,然后返回 f 数组中的最大值即可。动态规划转移方程:
本题 定义为以第 i 个数结尾的「连续子数组的最大和」,就是说计算时必须包括第 i 个数。
传统上,我们可能将 DP 数组定义为:dp[i]
表示“从 0 至 i 处的所有子数组中子数组元素之和的最大值”,然后最后将 dp[n-1]
返回出去就行。但在这个问题中其所求是子数组元素之和的最大值,它相当于对之前的多个元素有依赖关系,如果那样定义的话,则无法建立 dp[i+1]
和 dp[i]
之间的递推关系。
1 | // 普通的动态规划写法 空间复杂度为 O(n) |
复杂度:
- 时间复杂度:),其中
n
为nums
数组的长度。我们只需要遍历一遍数组即可求得答案。 - 空间复杂度:。我们只需要常数空间存放若干变量。
相关题目:
根据我的经验,使用动态规划算法时,如果满足以下条件:
- 题目只需要一个最终答案:
- 它可以是动态规划数组
dp
最后一个元素 - 或只是单纯需要知道的每一次
dp
计算结果(且只需了解一次)。比如本题。
- 它可以是动态规划数组
- 每一次计算
dp[i]
时,只需要了解前面常数的dp[i-1]
、dp[i-2]
等等。
那么就可以考虑压缩 dp
数组的长度,使空间复杂度为 ,而不是 。
Kadane’s Algorithm
Kadane’s Algorithm 这名字听起来确实挺高大上的。但是我有一个疑惑——所谓 Kadane’s Algorithm 就是指这种压缩空间的技巧吗?为此我收集了以下资料作为参考。
-
Kadane Algorithm | LeetCode The Hard Way
The Kadane’s algorithm is a well-known method for solving the problem of finding the maximum sum of a contiguous subarray of a given array of numbers. The basic idea behind the algorithm is to iterate through the array, keeping track of the maximum sum seen so far and the current sum, and updating the maximum sum whenever a new maximum is found. The algorithm has a time complexity of . -
动态规划 - 维基百科,自由的百科全书 (wikipedia.org)
使用动态规划的算法有:最长公共子序列、Floyd-Warshall 算法、Viterbi 算法、Kadane’s algorithm、求解马可夫决策过程下最佳策略、莱文斯坦距离。
(博主碎碎念:意思就是说 Kadane’s algorithm 是动态规划的一种思想。) -
知乎网友 @Vincent Fong
《论算法领域的俚语》:Kadane 算法、状压 dp、用滚动变量代替一维数组、动态规划之空间复杂度 O(n) 优化到 O(1)…其实说的都是同一个事情。
dp
数组压缩实践:把原本的 dp[i][j]
dp[i-1][j]
dp[i-2][j]
中的第一维变为模运算结果,即: dp[i%N][j]
dp[(i-1+M)%M][j]
dp[(i-2+K)%K][j]
如果运算时只需要用到上一行的变量,即 dp[i-1][j]
之类的,可尝试直接把第一维度删除:变为 dp[j]
,从而 dp
数组变为一维数组。
前缀和
本小节方法灵感来源于 站内文章「买卖股票的最佳时机」 这个题目,建议先了解一下题解中「一次遍历」方案。点击链接即可直达本站文章查看问题解法。
如果你已经理解「买卖股票的最佳时机」中一次遍历的算法思路,下面回到本文的「最大子数组问题」的前缀和解决思路:
- 计算数组的前缀和。比如
nums=[5,4,-1,7,8]
得到arr=[5,9,8,15,23]
。 - 原题目的求解变为:找出当 时,
arr[j]-arr[i]
最大的值。 - 和 121. 买卖股票的最佳时机 不同的是,本题需要考虑答案子数组长度不为 0,且长度可能拉满。
1 | public int maxSubArray(int[] nums) { |
连续子数组问题通常可以考虑使用前缀和的方法求解,详看:站内文章【刷题日记】学会前后缀分解,就能从「连续子数组」打到「接雨水」。
分治法之线段树
这个分治方法类似于「线段树求解最长公共上升子序列问题」的 pushUp
操作。
对于一个区间 [l,r]
,我们取 ,对区间 [l,m]
和 [m+1,r]
分治求解。当递归逐层深入直到区间长度缩小为 1 的时候,递归「开始回升」。这个时候我们考虑如何通过 [l,m]
区间的信息和 [m+1,r]
区间的信息合并成区间 [l,r]
的信息。
对于一个区间 [l,r]
,我们可以维护四个量:
lSum
表示[l,r]
内以l
为左端点的最大子段和rSum
表示[l,r]
内以r
为右端点的最大子段和mSum
表示[l,r]
内的最大子段和iSum
表示[l,r]
的区间和
以下简称 [l,m]
为 [l,r]
的「左子区间」,[m+1,r]
为 [l,r]
的「右子区间」。对于长度为 1 的区间 [i,i]
,四个量的值都和 nums[i]
相等。对于长度大于 1 的区间:
- 首先最好维护的是
iSum
,区间[l,r]
的iSum
就等于「左子区间」的iSum
加上「右子区间」的iSum
。 - 对于
[l,r]
的lSum
,存在两种可能,它要么等于「左子区间」的lSum
,要么等于「左子区间」的iSum
加上「右子区间」的lSum
,二者取大。 - 对于
[l,r]
的rSum
,同理,它要么等于「右子区间」的rSum
,要么等于「右子区间」的iSum
加上「左子区间」的rSum
,二者取大。 - 当计算好上面的三个量之后,就很好计算
[l,r]
的mSum
了。我们可以考虑[l,r]
的mSum
对应的区间是否跨越m
——它可能不跨越m
,也就是说[l,r]
的mSum
可能是「左子区间」的mSum
和 「右子区间」的mSum
中的一个;它也可能跨越m
,可能是「左子区间」的rSum
和 「右子区间」的lSum
求和。三者取大。
1 | class Solution { |
假设序列 a 的长度为 n。
复杂度:
- 时间复杂度:假设我们把递归的过程看作是一颗二叉树的先序遍历,那么这颗二叉树的深度的渐进上界为 ,这里的总时间相当于遍历这颗二叉树的所有节点,故总时间的渐进上界是 ,故渐进时间复杂度为 。
- 空间复杂度:递归会使用 的栈空间,故渐进空间复杂度为 。
「线段树方法」相较于「动态规划算法」来说,时间复杂度相同,但是因为使用了递归,并且维护了四个信息的结构体,运行的时间略长,空间复杂度也不如方法一优秀,而且难以理解。那么这种方法存在的意义是什么呢?
对于这道题而言,确实是如此的。但是仔细观察「线段树方法」,它不仅可以解决区间 [0,n-1]
,还可以用于解决任意的子区间 [l,r]
的问题。如果我们把 [0,n-1]
分治下去出现的所有子区间的信息都用堆式存储的方式记忆化下来,即建成一棵真正的树之后,我们就可以在 的时间内求到任意区间内的答案,我们甚至可以修改序列中的值,做一些简单的维护,之后仍然可以在 的时间内求到任意区间内的答案,对于大规模查询的情况下,这种方法的优势便体现了出来。这棵树就是一种神奇的数据结构——线段树。