题目描述
给定一个由整数数组 A
表示的**环形数组** C
,求 C
的非空子数组的最大可能和。
在此处,环形数组意味着数组的末端将会与开头相连呈环状。(形式上,当0 <= i < A.length
时 C[i] = A[i]
,而当 i >= 0
时 C[i+A.length] = C[i]
)
此外,子数组最多只能包含固定缓冲区 A
中的每个元素一次。(形式上,对于子数组 C[i], C[i+1], …, C[j]
,不存在 i <= k1, k2 <= j
其中 k1 % A.length = k2 % A.length
)
示例 1:
输入:[1,-2,3,-2] 输出:3 解释:从子数组 [3] 得到最大和 3
示例 2:
输入:[5,-3,5] 输出:10 解释:从子数组 [5,5] 得到最大和 5 + 5 = 10
示例 3:
输入:[3,-1,2,-1] 输出:4 解释:从子数组 [2,-1,3] 得到最大和 2 + (-1) + 3 = 4
示例 4:
输入:[3,-2,2,-3] 输出:3 解释:从子数组 [3] 和 [3,-2,2] 都可以得到最大和 3
示例 5:
输入:[-2,-3,-1] 输出:-1 解释:从子数组 [-1] 得到最大和 -1
提示:
1. -30000 <= A[i] <= 30000
2. 1 <= A.length <= 30000
解题思路:
这题结果只有两种情况。一种是最大子序和没有跨越边界。另一种是最大子序和跨越了边界
对于没有跨越边界的情况,我们只需当作不是环形数组来解决。而跨越了边界的情况,我们可以用数组的和减去不是环形数组时的最小子序和的方法来求。最后比较两种情况的大小就可以了。
代码如下:
public int maxSubarraySumCircular(int[] A) {
if (A.length == 1) return A[0];
int max = A[0]; //求A的最大字序和
int min = A[0]; //求A的最小字序和
int sum = A[0]; //A的和
int maxM = A[0]; //当前最大字序和
int minM = A[0]; //当前最小字序和
for (int i = 1; i < A.length; i ++) {
sum += A[i];
maxM = Math.max(maxM + A[i], A[i]);
minM = Math.min(minM + A[i], A[i]);
if (maxM > max)
max = maxM;
if (minM < min)
min = minM;
}
if (max < 0)
return max;
return Math.max(max, sum - min);
}
顺便贴上自己刚建的博客地址:http://rikin04.club/2019/04/19/Leetcode918. 环形子数组的最大和/