-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathProblem_0918_maxSubarraySumCircular.cc
121 lines (112 loc) · 3.3 KB
/
Problem_0918_maxSubarraySumCircular.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
#include <deque>
#include <vector>
#include "UnitTest.h"
using namespace std;
// @sa https://www.bilibili.com/video/BV1pw411M7Du/
class Solution
{
public:
int maxSubarraySumCircular1(vector<int>& nums)
{
int n = nums.size();
// leftMax[i] 表示必须以arr[0]开始, [0, i] 区间内的最大子数组和
vector<int> leftMax(n);
leftMax[0] = nums[0];
int leftSum = nums[0];
// pre 则表示的是必须以 arr[i] 为结尾的最大子数组和
int pre = nums[0];
int ans = nums[0];
for (int i = 1; i < n; i++)
{
pre = std::max(pre + nums[i], nums[i]);
ans = std::max(ans, pre);
leftSum += nums[i];
leftMax[i] = std::max(leftMax[i - 1], leftSum);
}
int rightSum = 0;
for (int i = n - 1; i > 0; i--)
{
// 固定右侧的区间,在左区间找最大的子数组和
rightSum += nums[i];
ans = std::max(ans, rightSum + leftMax[i - 1]);
}
return ans;
}
// 可以考虑成两种情况,
// 情况一:是连续的数组吗,那么直接求最大值
// 情况二:非连续的,那么一定是开头一段 + 结尾一段的最大值
// 我们反向考虑,求开头一段 + 结尾一段的最大值 = 数组所有元素和 - 中间一段的最小值
int maxSubarraySumCircular2(vector<int>& nums)
{
int n = nums.size();
// preMax 表示必须以 nums[i] 为结尾子数组的最大值
int preMax = nums[0];
int max = nums[0];
// preMin 表示必须以 nums[i] 为结尾子数组的最小值
int preMin = nums[0];
int min = nums[0];
int sum = nums[0];
for (int i = 1; i < n; i++)
{
preMax = std::max(preMax + nums[i], nums[i]);
max = std::max(max, preMax);
preMin = std::min(preMin + nums[i], nums[i]);
min = std::min(min, preMin);
sum += nums[i];
}
// [-5, -3, -1]
// sum = -9
// max = -1
// min = -9
// 中间段的数组最小值是 -9 那么会导致左右两端没有元素
// 明显不符合题设条件
return sum == min ? max : std::max(max, sum - min);
}
int maxSubarraySumCircular3(vector<int>& nums)
{
int n = nums.size();
deque<pair<int, int>> q;
int pre = nums[0];
int ans = nums[0];
q.push_back({0, pre});
// 数组延长一倍,延长的部分用原数组填充
for (int i = 1; i < 2 * n; i++)
{
while (!q.empty() && q.front().first < i - n)
{
// 队列 [j ... i]
// 保证队列的长度不超过 n ,即 i - j <= n
q.pop_front();
}
// 前缀和
pre += nums[i % n];
// 统计 [j,i] 区间和的最大值
ans = std::max(ans, pre - q.front().second);
// TODO: 下面的 while 删掉为什么会出错?
while (!q.empty() && q.back().second >= pre)
{
q.pop_back();
}
q.push_back({i, pre});
}
return ans;
}
};
void test()
{
Solution s;
vector<int> n1 = {1, -2, 3, -2};
vector<int> n2 = {5, -3, 5};
EXPECT_EQ_INT(3, s.maxSubarraySumCircular1(n1));
EXPECT_EQ_INT(10, s.maxSubarraySumCircular1(n2));
EXPECT_EQ_INT(3, s.maxSubarraySumCircular2(n1));
EXPECT_EQ_INT(10, s.maxSubarraySumCircular2(n2));
EXPECT_EQ_INT(3, s.maxSubarraySumCircular3(n1));
EXPECT_EQ_INT(10, s.maxSubarraySumCircular3(n2));
EXPECT_SUMMARY;
}
int main()
{
test();
return 0;
}