42. 最大子数组 II

描述

给定一个整数数组,找出两个不重叠子数组使得它们的和最大。每个子数组的数字在数组中的位置应该是连续的。返回最大的和。

注意事项

子数组最少包含一个数

样例

给出数组
[1, 3, -1, 2, -1, 2]这两个子数组分别为
[1, 3][2, -1, 2]或者[1, 3, -1, 2][2],它们的最大和都是
7

挑战
要求时间复杂度为 O(n)

思路

这个题的思路是,因为 两个subarray 一定不重叠,所以必定存在一条分割线,分开这两个 subarrays,所以最后的部分里:

  max = Integer.MIN_VALUE;
        for(int i = 0; i < size - 1; i++){
            max = Math.max(max, left[i] + right[i + 1]);
        }
        return max;

这里是在枚举 这条分割线的位置,然后 left[] 和 right[] 里分别存的是,某个位置往左的 maximum subarray 和往右的 maximum subarray

代码

public class Solution {
    /**
     * @param nums: A list of integers
     * @return: An integer denotes the sum of max two non-overlapping subarrays
     */
    public int maxTwoSubArrays(ArrayList<Integer> nums) {
        int size = nums.size();
        int[] left = new int[size];
        int[] right = new int[size];
        
        // 当前i往左的最大子数组和
        int sum = 0;
        int minSum = 0;
        int max = Integer.MIN_VALUE;
        for (int i = 0; i < size; i++) {
            sum += nums.get(i);
            max = Math.max(max, sum - minSum);
            minSum = Math.min(sum, minSum);
            left[i] = max;
        }
        
        // 当前i往右的最大子数组和
        sum = 0;
        minSum = 0;
        max = Integer.MIN_VALUE;
        for (int i = size - 1; i >= 0; i--) {
            sum += nums.get(i);
            max = Math.max(max, sum - minSum);
            minSum = Math.min(sum, minSum);
            right[i] = max;
        }
        
        max = Integer.MIN_VALUE;
        // left[i]和right[i + 1]之间存在分割线
        // right是i + 1,i 最大取size - 2,不能越界
        for (int i = 0; i < size - 1; i++) {
            max = Math.max(max, left[i] + right[i + 1]);
        }
        return max;
    }
}

总结

关于前缀序列求和,值得注意的是,比如当前 i 的max是5,若i + 1的sum小于max则i + 1的max仍为5,即对于left数组,left[i] = 5,
left[i + 1] = 5

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 背景 一年多以前我在知乎上答了有关LeetCode的问题, 分享了一些自己做题目的经验。 张土汪:刷leetcod...
    土汪阅读 12,769评论 0 33
  • Maximum Subarray 由于简书不支持latex语法,所以可以到下面的作业部落去看。https://ww...
    别时茫茫阅读 2,392评论 0 2
  • 分治策略 本文包括分治的基本概念二分查找快速排序归并排序找出伪币棋盘覆盖最大子数组 源码链接:https://gi...
    廖少少阅读 1,893评论 0 7
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,765评论 18 399
  • 我一口气读完《女人是一场修炼》这本并不算长的回忆录,中间曾经几度恍惚,儿时、中学、大学时代的种种记忆犹如潮水般涌...
    lily123321阅读 587评论 1 3