剑指offer-连续子数组的最大和(JavaScript实现)

作者 : 开心源码 本文共2801个字,预计阅读时间需要8分钟 发布时间: 2022-05-13 共230人阅读

尽管这题在 leetcode 上标注的是「简单」难度,但是解法有 4 种,并且都非常具备代表性。比较容易想到的是基础的动态规划法。

解法 1:动态规划

定义状态数组dp[i]的含义:数组中元素下标为[0, i]的连续子数组最大和。

状态转移的过程如下:

  • 初始情况:dp[0] = nums[0]
  • nums[i] > 0,那么 dp[i] = nums[i] + dp[i - 1]
  • nums[i] <= 0,那么 dp[i] = nums[i]

代码实现如下:

// 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)

解法 2:原地进行动态规划

解法 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)

解法 3:贪心法

贪心法的题目比较少见,而且一般都比较难证实。本题的贪心法的思路是:在循环中找到不断找到当前最优的和 sum。

注意:sum 是 nums[i]sum + nums[i]中最大的值。这种做法保证了 sum 是一直是针对连续数组算和。

代码实现如下:

// 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)

解法 4: 分治法

分治法的做题思路是:先将问题分解为子问题;处理子问题后,再将子问题合并,处理主问题。

使用分治法解本题的思路是:

  • 将数组分为 2 部分。例如 [1, 2, 3, 4] 被分为 [1, 2][3, 4]
  • 通过递归计算,得到左右两部分的最大子序列和是 lsum,rsum
  • 从数组中间开始向两边计算最大子序列和 cross
  • 返回 max(lsum, cross, rsum)

整体过程可以参考来自 Leetcode 官方题解的图:

image

可以看到,分治法可行的关键的是:最大子序列和只可能出现在左子数组、右子数组或者横跨左右子数组 这三种情况下。

代码实现如下:

// 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)

更多资料

整理不易,若对您有帮助,请给个「关注+点赞」,您的支持是我升级的动力 ??

  • ??Blog:剑指 Offer 题解 + JS 代码
  • ??Github : dongyuanxin/blog
  • ?? 公众号:心谭博客
说明
1. 本站所有资源来源于用户上传和网络,如有侵权请邮件联系站长!
2. 分享目的仅供大家学习和交流,您必须在下载后24小时内删除!
3. 不得使用于非法商业用途,不得违反国家法律。否则后果自负!
4. 本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解!
5. 如有链接无法下载、失效或广告,请联系管理员处理!
6. 本站资源售价只是摆设,本站源码仅提供给会员学习使用!
7. 如遇到加密压缩包,请使用360解压,如遇到无法解压的请联系管理员
开心源码网 » 剑指offer-连续子数组的最大和(JavaScript实现)

发表回复