本篇博客是我对“水果成篮”这道题由暴力解法到滑动窗口思路的具体思路,有需要借鉴即可。
目录
- 1.题目
- 2.暴力求解
- 3.暴力优化
- 3.1每次right不用回退
- 3.2有些left长度一定不如前一个,不用走,left不回退
- 4.滑动窗口算法
- 5.总结
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