虽然这题在 leetcode 上标注的是「简单」难度,可是解法有 4 种,而且都很是具备表明性。比较容易想到的是基础的动态规划法。javascript
定义状态数组dp[i]
的含义:数组中元素下标为[0, i]
的连续子数组最大和。java
状态转移的过程以下:git
dp[0] = nums[0]
nums[i] > 0
,那么 dp[i] = nums[i] + dp[i - 1]
nums[i] <= 0
,那么 dp[i] = nums[i]
代码实现以下:github
// ac地址:https://leetcode-cn.com/problems/maximum-subarray/ // 原文地址:https://xxoo521.com/2020-03-09-max-sub-sum/ /** * @param {number[]} nums * @return {number} */ var maxSubArray = function(nums) { const dp = []; let res = (dp[0] = nums[0]); for (let i = 1; i < nums.length; ++i) { dp[i] = nums[i]; if (dp[i - 1] > 0) { dp[i] += dp[i - 1]; } res = Math.max(res, dp[i]); } return res; };
时间复杂度和空间复杂度都是\(O(N)\)。数组
解法 1 中开辟了 dp 数组。其实在原数组上作修改,用nums[i]
来表示dp[i]
。因此解法 1 的代码能够优化为:优化
// ac地址:https://leetcode-cn.com/problems/maximum-subarray/ // 原文地址:https://xxoo521.com/2020-03-09-max-sub-sum/ /** * @param {number[]} nums * @return {number} */ var maxSubArray = function(nums) { let res = nums[0]; for (let i = 1; i < nums.length; ++i) { if (nums[i - 1] > 0) { nums[i] += nums[i - 1]; } res = Math.max(res, nums[i]); } return res; };
不用开辟额外空间,因此空间复杂度降为\(O(1)\)。ui
贪心法的题目比较少见,并且通常都比较难证实。本题的贪心法的思路是:在循环中找到不断找到当前最优的和 sum。spa
注意:sum 是 nums[i]
和 sum + nums[i]
中最大的值。这种作法保证了 sum 是一直是针对连续数组算和。code
代码实现以下:blog
// ac地址:https://leetcode-cn.com/problems/maximum-subarray/ // 原文地址:https://xxoo521.com/2020-03-09-max-sub-sum/ /** * @param {number[]} nums * @return {number} */ var maxSubArray = function(nums) { let maxSum = (sum = nums[0]); for (let i = 1; i < nums.length; ++i) { sum = Math.max(nums[i], sum + nums[i]); maxSum = Math.max(maxSum, sum); } return maxSum; };
空间复杂度为\(O(1)\),时间复杂度为\(O(N)\)
分治法的作题思路是:先将问题分解为子问题;解决子问题后,再将子问题合并,解决主问题。
使用分治法解本题的思路是:
[1, 2, 3, 4]
被分为 [1, 2]
和 [3, 4]
总体过程能够参考来自 Leetcode 官方题解的图:
能够看到,分治法可行的关键的是:最大子序列和只可能出如今左子数组、右子数组或横跨左右子数组 这三种状况下。
代码实现以下:
// ac地址:https://leetcode-cn.com/problems/maximum-subarray/ // 原文地址:https://xxoo521.com/2020-03-09-max-sub-sum/ /** * @param {number[]} nums * @param {number} left * @param {number} right * @param {number} mid * @return {number} */ function crossSum(nums, left, right, mid) { if (left === right) { return nums[left]; } let leftMaxSum = Number.MIN_SAFE_INTEGER; let leftSum = 0; for (let i = mid; i >= left; --i) { leftSum += nums[i]; leftMaxSum = Math.max(leftMaxSum, leftSum); } let rightMaxSum = Number.MIN_SAFE_INTEGER; let rightSum = 0; for (let i = mid + 1; i <= right; ++i) { rightSum += nums[i]; rightMaxSum = Math.max(rightMaxSum, rightSum); } return leftMaxSum + rightMaxSum; } /** * @param {number[]} nums * @param {number} left * @param {number} right * @return {number} */ function __maxSubArray(nums, left, right) { if (left === right) { return nums[left]; } const mid = Math.floor((left + right) / 2); const lsum = __maxSubArray(nums, left, mid); const rsum = __maxSubArray(nums, mid + 1, right); const cross = crossSum(nums, left, right, mid); return Math.max(lsum, rsum, cross); } /** * @param {number[]} nums * @return {number} */ var maxSubArray = function(nums) { return __maxSubArray(nums, 0, nums.length - 1); };
时间复杂度是\(O(NlogN)\)。因为递归调用,因此空间复杂度是\(O(logN)\)
整理不易,若对您有帮助,请给个「关注+点赞」,您的支持是我更新的动力 👇