模块二——滑动窗口:最大连续1的个数 |||
题目解析
二进制数组代表给我们的数组里面只有0和1,题目中的翻转其实我们看示例1可以得到是将0变成1的过程,这里最需要注意的就是最多,也就是我们不一定要翻转k个0,返回值就是这个连续1的最大长度。可以参考示例1,最大长度即为6。
算法原理
这道题直接做是很恶心的,因为你去枚举第一种情况就翻转一次,去枚举其他情况又要翻转回来,所以我们需要将直接枚举的方式等价处理为在原始数组中找出最长的子数组,0的个数不超过k个,因为子数组中的0的个数不超过k个,所以我们一定能够翻转成功。
解法一:暴力枚举(超时)+ zero计数器(int 类型的变量)
固定一个left起点,利用right枚举每一个子数组,以及zero计数器去寻找0的个数不超过k个的最长的子数组即可。
解法二:滑动窗口(时间复杂度为O(n),空间复杂度为O(1))
我们可以发现在暴力枚举中,我们每找到一个合法子数组都需要让right倒回到left++的位置,但这却是没有必要的,因为[left,right-1]这个区间已经是不超过k的最优子数组了,所以我们直接让right不动,让left越过一段区间,直到zero小于k。这就保证了我们可以使用滑动窗口来解决这道题了,接下来就是走模板的老套路。
1.int left = 0,right = 0;//窗口左右端点
2.if(nums[right] == 0)zero++;//进窗口
3. while(zero > k) if(nums[left++] == 0)zero--;//判断and出窗口
4.len = max(len,right - left + 1);//更新结果
代码编写
解法一:暴力枚举(超时)+ zero计数器(int 类型的变量)
class Solution {
public:
int longestOnes(vector<int>& nums, int k) {
int len = 0,n = nums.size();
for(int left = 0;left < n;left++)
{
int zero = 0;
for(int right = left;right < n;right++)
{
if(nums[right] == 0)zero++;
if(zero > k)break;//前面重点提醒的最多就体现在这一步和下一步,因为在这里踩过坑就多说两句,我们right边走边判断,然后再更新结果,而不是把更新结果放判断里面
len = max(len,right - left + 1);
}
}
return len;
}
};
解法二:滑动窗口(时间复杂度为O(n),空间复杂度为O(1))
class Solution {
public:
int longestOnes(vector<int>& nums, int k) {
int len = 0;
for(int left = 0,right = 0,zero = 0;right < nums.size();right++)//窗口的左右端点
{
if(nums[right] == 0)zero++;//进窗口
while(zero > k) if(nums[left++] == 0)zero--;//判断and出窗口
len = max(len,right - left + 1);//更新结果
}
return len;
}
};