题目
水果成篮
在一排树中,第 i 棵树产生 tree[i] 型的水果。
你可以从你选择的任何树开始,然后重复执行以下步骤:
把这棵树上的水果放进你的篮子里。如果你做不到,就停下来。
移动到当前树右侧的下一棵树。如果右边没有树,就停下来。
请注意,在选择一颗树后,你没有任何选择:你必须执行步骤 1,然后执行步骤 2,然后返回步骤 1,然后执行步骤 2,依此类推,直至停止。
你有两个篮子,每个篮子可以携带任何数量的水果,但你希望每个篮子只携带一种类型的水果。
用这个程序你能收集的水果总量是多少?
示例 1:
输入:[1,2,1]
输出:3
解释:我们可以收集 [1,2,1]。
示例 2:
输入:[0,1,2,2]
输出:3
解释:我们可以收集 [1,2,2].
如果我们从第一棵树开始,我们将只能收集到 [0, 1]。
示例 3:
输入:[1,2,3,2,2]
输出:4
解释:我们可以收集 [2,3,2,2].
如果我们从第一棵树开始,我们将只能收集到 [1, 2]。
示例 4:
输入:[3,3,3,1,2,1,1,2,3,3,4]
输出:5
解释:我们可以收集 [1,2,1,1,2].
如果我们从第一棵树或第八棵树开始,我们将只能收集到 4 个水果。
提示:
1 <= tree.length <= 40000
0 <= tree[i] < tree.length
思路
- 这里我用begin表示最开始的位置,first和second指代两种水果(没有顺序)
- 然后map用它们当key,value存储的是它们最接近 i 时第一次出现的位置,方便迭代begin。
- 比如 1,2,1,1,3,3,1, 当 i 为3的时候,存储的 map[1]=2,map[2]=1,然后当 i 遍历到4的时候,begin=map[1]=2,并把key为2的水果删除,增加key为3的水果,并记录位置。
代码
class Solution {
public:
int totalFruit(vector<int>& tree) {
int max = 0, first, second, begin;
map<int, int> map;
for (int i = 0; i < tree.size(); i++) {
if (map.find(tree[i]) == map.end()) {
if (map.empty()) {
first = tree[i];
begin = i;
}
else if (map.size() == 1)
second = tree[i];
else {
// 出现新的水果,替换离i最远的水果
if (tree[i - 1] == first) {
map.erase(second);
second = tree[i];
begin = map[first];
}
else {
map.erase(first);
first = tree[i];
begin = map[second];
}
}
map.insert(make_pair(tree[i], i));
}
else {
// 存储其最接近i的第一次出现的位置
if (tree[i - 1] != tree[i]) {
map[tree[i]] = i;
}
}
// 每次迭代判断更新max
if (!map.empty() && max < i - begin + 1)
max = i - begin + 1;
}
return max;
}
};
翻了下前面排名的代码,简练,羡慕:
int cnt[40005];
class Solution {
public:
int totalFruit(vector<int>& tree) {
int n = (int) tree.size();
memset(cnt, 0, n * sizeof(int));
int j = 0;
int types = 0;
int ans = 0;
for (int i = 0; i < n; i++) {
int x = tree[i];
if (++cnt[x] == 1) {
++types;
}
while (types > 2) {
int y = tree[j++];
if (--cnt[y] == 0) {
--types;
}
}
ans = max(ans, i - j + 1);
}
return ans;
}
};