水果成蓝

题目

水果成篮
在一排树中,第 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

思路

  1. 这里我用begin表示最开始的位置,first和second指代两种水果(没有顺序)
  2. 然后map用它们当key,value存储的是它们最接近 i 时第一次出现的位置,方便迭代begin。
  3. 比如 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;
    }
};
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值