动态规划:子数组的最大和

本文介绍了LeetCode上的两道题目,分别是53号题(最大子数组和)和918号题(环形子数组的最大和)。对于最大子数组和,采用前缀和策略,通过比较当前元素与前缀和之和来更新最大和。而对于环形子数组的最大和,首先尝试不含环的情况,然后寻找可能存在的负数环,通过减去最小子序列和来得到最大和。文章提供了两种思路的代码实现,但第二种对于环形数组的优化思路在某些情况下可能导致超时。
摘要由CSDN通过智能技术生成

简介

本文主要介绍了一个系列的问题:子数组的最大和环形子数组的最大和。问题来自LeetCode。

LC:53 子数组和问题

给定一个整数数组 nums ,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。

这个思路较为简单,考虑计算每个位置的前缀和:preSum,在遍历到下一个位置时,比较preSum+curNum是否比curSum大,如果大则 preSum+=n,否则preSum=n。
每次比较完之后,需要记录下最大值。
这样做其实是一种贪心的策略,如果curNum加上该位置之前的前缀和反而变小了,那么最大和的子序列应该从当前位置重新开始。

class Solution {
    public int maxSubArray(int[] nums) {
        int max = nums[0], preSum = nums[0];
        for (int i = 1; i < nums.length; i++) {
            int n = nums[i];
            preSum = Math.max(preSum+n, n);
            max = max > preSum ? max : preSum;
        }
        return max;
    }
}

LC 918. 环形子数组的最大和

在这里插入图片描述

1. 简单思路

在这里插入图片描述

class Solution {
    public static int maxSubarraySumCircular(int[] nums) {
        int len = nums.length;
        int preSum = 0, maxSum = nums[0];
        for (int i = 0; i < len; i++) {
            preSum = nums[i];
            for (int j = i+1; j < len + i; j++) {
                int idx = j % len;
                preSum = Math.max(preSum + nums[idx], nums[idx]);
                maxSum = Math.max(preSum, maxSum);
            }
        }
        return maxSum;
    }
}

结果会超时。

优化思路

在这里插入图片描述

class Solution {
    public static int maxSubarraySumCircular(int[] nums) {
        int preSum = 0, maxSum = nums[0], sum = 0, minSum = 0;
        // 最优解中不涉及环
        for (int i = 0; i < nums.length; i++) {
            int n = nums[i];
            sum += n;
            preSum = Math.max(preSum + n, n);
            maxSum = Math.max(preSum, maxSum);
        }
        // 如果有环,则说明在数组内部有一个最长的最小子序列,把它找出并去掉(找出负数)
        preSum = 0;
        for (int i = 1; i < nums.length-1; i++) {
            preSum = Math.min(preSum + nums[i], 0);
            minSum = Math.min(preSum, minSum);
        }
        return Math.max(sum-minSum, maxSum);
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值