53、最大子陣列和 | 演算法(leetcode,附思維導圖 + 全部解法)300題

碼農三少發表於2022-05-28

零 標題:演算法(leetcode,附思維導圖 + 全部解法)300題之(53)最大子陣列和

一 題目描述

題目描述

二 解法總覽(思維導圖)

思維導圖

三 全部解法

1 方案1

1)程式碼:

// 解法1 “自己。貪心法”。

// 思路:
// 1)狀態初始化 l = nums.length; sum = 0, resMaxVal = Number.NEGATIVE_INFINITY; 。
// 2)核心:遍歷陣列。
// 2.1)核心:若 此時 sum < 0,說明我還不如從0開始 —— 即重置sum為0。
// 2.2)sum 加上當前值 num[i] 。
// 2.3)根據 sum 情況,更新 resMaxVal 值。
// 3)返回結果 resMaxVal 。
var maxSubArray = function(nums) {
    // 1)狀態初始化 l = nums.length; sum = 0, resMaxVal = Number.NEGATIVE_INFINITY; 。
    const l = nums.length;
    let sum = 0,
        resMaxVal = Number.NEGATIVE_INFINITY;

    // 2)核心:遍歷陣列。
    for (let i = 0; i < l; i++) {
        const tempVal = nums[i];
        // 2.1)核心:若 此時 sum < 0,說明我還不如從0開始 —— 即重置sum為0。
        if (sum < 0) {
            sum = 0;
        }
        // 2.2)sum 加上當前值 num[i] 。
        sum += tempVal;
        // 2.3)根據 sum 情況,更新 resMaxVal 值。
        resMaxVal = Math.max(resMaxVal, sum);
    }

    // 3)返回結果 resMaxVal 。
    return resMaxVal;
}

2 方案2

1)程式碼:

// 解法2 “動態規劃法”。
// 參考:
// 1)https://leetcode.cn/problems/maximum-subarray/solution/zui-da-zi-xu-he-by-leetcode-solution/

// 思路:
// 狀態定義:dp[i] 表示以位置 i 結尾的最大子陣列和。
// 狀態轉移:dp[i] = max(nums[i], nums[i] + dp[i - 1])。
// 1)狀態初始化 l = nums.length; dp = [nums[0]]; 。
// 2)核心:狀態轉移。
// 3)dp裡的最大值就是答案 —— resMaxVal 。
// 4)返回結果 resMaxVal 。
var maxSubArray = function(nums) {
    // 1)狀態初始化 l = nums.length; dp = [nums[0]]; 。
    const l = nums.length;
    let dp = [nums[0]];

    // 2)核心:狀態轉移。
    for (let i = 1 ; i < l; i++) {
        const tempVal = nums[i];
        dp[i] = Math.max(tempVal, tempVal + dp[i - 1]);
    }

    // 3)dp裡的最大值就是答案 —— resMaxVal 。
    const resMaxVal = Math.max(...dp);

    // 4)返回結果 resMaxVal 。
    return resMaxVal;
}

3 方案3

1)程式碼:

// 方案3 “分治法”。
// 注:“似乎時間、空間複雜度都不是很好~”。
// 參考:
// 1)https://leetcode.cn/problems/maximum-subarray/solution/zui-da-zi-xu-he-by-leetcode-solution/

// 思路:
// 暫略(TODO)。
var maxSubArray = function(nums) {
    const Status = function(l, r, m, i) {
        this.lSum = l;
        this.rSum = r;
        this.mSum = m;
        this.iSum = i;
    }

    const pushUp = (l, r) => {
        const iSum = l.iSum + r.iSum;
        const lSum = Math.max(l.lSum, l.iSum + r.lSum);
        const rSum = Math.max(r.rSum, r.iSum + l.rSum);
        const mSum = Math.max(Math.max(l.mSum, r.mSum), l.rSum + r.lSum);
        return new Status(lSum, rSum, mSum, iSum);
    }

    const getInfo = (a, l, r) => {
        if (l === r) {
            return new Status(a[l], a[l], a[l], a[l]);
        }
        const m = (l + r) >> 1;
        const lSub = getInfo(a, l, m);
        const rSub = getInfo(a, m + 1, r);
        return pushUp(lSub, rSub);
    }

    return getInfo(nums, 0, nums.length - 1).mSum;
}

四 資源分享 & 更多

1 歷史文章 - 總覽

歷史文章 - 總覽

刷題進度 - LeetCode:511 / 2648 、《劍指offer》:66 / 66

2 博主簡介

碼農三少 ,一個致力於編寫 極簡、但齊全題解(演算法) 的博主。
專注於 一題多解、結構化思維 ,歡迎一起刷穿 LeetCode ~

相關文章