【算法】滑动窗口——水果成篮

本篇博客是我对“水果成篮”这道题由暴力解法到滑动窗口思路的具体思路,有需要借鉴即可。

1.题目

题目链接:LINK

在这里插入图片描述

这道题,题干很长,大概意思是给你一个数组,然后你可以从随便一个数字开始依次遍历,直到遍历结束或者找到超过不同的两个数字,问你最大长度是几。

一般没有经验的人,肯定首先想到的是暴力求解,我也是一样哈,没啥经验。

2.暴力求解

那如何进行暴力求解呢?
暴力求解思路:哈希表 + 依次遍历

思路详解:具体来说,定义两个指针,一个left,一个right,再搞一个哈希表,right指向的数字就丢到哈希表里,计数,然后超过两个或者遍历结束的时候,我们更新一下结果然后left++,right = left,重新计数。然后这堆记的数字取个最大的就行了。

总而言之,代码如下:

class Solution {
public:
//暴力解法:
    int totalFruit(vector<int>& fruits) {
        int n = fruits.size();
        int count = 0;
        int ret = 0;
        int sum = 0;
        for(int left = 0;left < n;left++)
        {
            count = 0;
            sum = 0;
            int hash[100001] = {0};
            for(int right = left;right < n;right++)
            {
                
                //进哈希表
                if(hash[fruits[right]] == 0)//一个新种类的水果
                {
                    hash[fruits[right]]++;
                    count++;
                    sum++;
                }
                else//这里标识之前有过的一个水果
                {
                    hash[fruits[right]]++;
                    sum++;
                }

                if(count > 2)
                {
                     ret = max(ret, sum - 1);break;
                }
               
                if(count <= 2 && right == n - 1)
                {ret = max(ret, sum);break;}
            }
        }

        return ret;
    }
};

然后不出意外哈,力扣绝对过不了,不然这题不会上中等难度了。

然后力扣提示说超出时间限制,比如下图:
在这里插入图片描述

所以说,这道题肯定是可以优化的,那具体怎么优化呢?我们一步一步来。

3.暴力优化

3.1每次right不用回退

这是什么意思呢?如下图所示:
在这里插入图片描述
在left固定的情况下,之后right这个地方数字种类超过2了,那么意思是红色这块区域数字种类个数是2,这时候按照暴力求解的方式,left++,right = left。

但是,我想说绿色的这块区域的数字种类会有可能超过2吗?显然这是不可能的,所以说right压根没有回去的必要。

3.2有些left长度一定不如前一个,不用走,left不回退

再比如说哈,如下图所示:
在这里插入图片描述
left = 2的情况下压根没必要去走好吧~~
在这里插入图片描述

行了,这时候我就发现这个题目经过优化之后满足“滑动窗口”的使用条件,两指针同向且不回退。

4.滑动窗口算法

然后可以直接优化成滑动窗口算法

class Solution 
{
public:
//滑动窗口:
    int totalFruit(vector<int>& fruits) {

        int n = fruits.size(), ret = 0,sum = 0,kands = 0;
        int hash[100001] = {0};//哈希数组

        for(int right = 0, left = 0;right < n;right++)
        {
            //进窗口
            if(hash[fruits[right]] == 0)kands++;//如果这个是一个新来的数,那么就种类++
            hash[fruits[right]]++;    
            sum++; 

            //出窗口
            while(kands > 2)
            {
                hash[fruits[left]]--;
                sum--;
                if(hash[fruits[left]] == 0)kands--;
                left++;
            }
           
           ret = max(ret, sum);
        }

        return ret;
    }
};

5.总结

总结一下这个题,我感觉正常做的话能做出来,也没啥坑,我感觉是属于一个很常规的滑动窗口的题目。


EOF

  • 27
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值