当前位置: 首页 > article >正文

[算法初阶]第二集 滑动窗口

大家好啊,好久没有更新了,最近比较忙,所以来更新初阶算法,正好复习一下,感谢大家的观看,如有错误欢迎指出。


下面我们来看题目吧!


1.209. 长度最小的子数组

这题大家想必一眼就看出了解法一暴力法

这个解法很简单

代码如下,不做多的解释

class Solution {
public:
    int minSubArrayLen(int s, vector<int>& nums) 
    {
        int n = nums.size();
        if (n == 0) 
        {
            return 0;
        }
        int ans = INT_MAX;
        for (int i = 0; i < n; i++) 
        {
            int sum = 0;
            for (int j = i; j < n; j++) 
            {
                sum += nums[j];
                if (sum >= s) 
                {
                    ans = min(ans, j - i + 1);
                    break;
                }
            }
        }
        return ans == INT_MAX ? 0 : ans;
    }
};

这个解法时间复杂度是O(n^{2})其中 n 是数组的长度。需要遍历每个下标作为子数组的开始下标,对于每个开始下标,需要遍历其后面的下标得到长度最小的子数组。

但是这不是我们要学的思路,我们要探究的是另一种的解法——滑动窗口

该解法思路:

定义两个指针 start 和 end 分别表示子数组(滑动窗口窗口)的开始位置和结束位置,维护变量 sum 存储子数组中的元素和(即从 nums[start] 到 nums[end] 的元素和)。

初始状态下,start 和 end 都指向下标 0,sum 的值为 0。

每一轮迭代,将 nums[end] 加到 sum,如果 sum≥s,则更新子数组的最小长度(此时子数组的长度是 end−start+1),然后将 nums[start] 从 sum 中减去并将 start 右移,直到 sum<s,在此过程中同样更新子数组的最小长度。在每一轮迭代的最后,将 end 右移。

代码如下

class Solution {
public:
    int minSubArrayLen(int s, vector<int>& nums) 
    {
        int n = nums.size();
        if (n == 0) 
        {
            return 0;
        }
        int ans = INT_MAX;
        int start = 0, end = 0;
        int sum = 0;
        while (end < n) 
        {
            sum += nums[end];
            while (sum >= s) 
            {
                ans = min(ans, end - start + 1);
                sum -= nums[start];
                start++;
            }
            end++;
        }
        return ans == INT_MAX ? 0 : ans;
    }
};

该方法时间复杂度:O(n)

2.3. 无重复字符的最长子串 - 力扣(LeetCode)

解法⼀(暴⼒求解,可以通过):
算法思路:
枚举「从每⼀个位置」开始往后,⽆重复字符的子串可以到达什么位置。找出其中⻓度最⼤的即
可。 在往后寻找⽆重复⼦串能到达的位置时,可以用「哈希表」统计出字符出现的频次
class Solution {
public:
 int lengthOfLongestSubstring(string s) {
 int ret = 0; // 记录结果
 int n = s.length();
 // 1. 枚举从不同位置开始的最⻓重复⼦串
 // 枚举起始位置
 for (int i = 0; i < n; i++)
 {
// 创建⼀个哈希表,统计频次
 int hash[128] = { 0 };
 
 // 寻找结束为⽌
 for (int j = i; j < n; j++)
 {
 hash[s[j]]++; // 统计字符出现的频次
 if (hash[s[j]] > 1) // 如果出现重复的
 break;
 
 // 如果没有重复,就更新 ret
 ret = max(ret, j - i + 1);
 }
 }
 // 2. 返回结果
 return ret;
 }
};

当然,我们知道我们想要的也不是这种解法

解法⼆(滑动窗口):
算法思路:
研究的对象依旧是⼀段连续的区间,因此继续使⽤「滑动窗口」思想来优化。
让滑动窗口满⾜:窗口内所有元素都是不重复的。
做法:右端元素 ch 进⼊窗口的时候,哈希表统计这个字符的频次:
  • 如果这个字符出现的频次超过 1 ,说明窗⼝内有重复元素,那么就从左侧开始划出窗口,直到 ch 这个元素的频次变为 1 ,然后再更新结果。
  • 如果没有超过 1 ,说明当前窗⼝没有重复元素,可以直接更新结果
    class Solution
    {
    public:
     int lengthOfLongestSubstring(string s) 
     {
     int hash[128] = { 0 }; // 使⽤数组来模拟哈希表
     int left = 0, right = 0, n = s.size();
     int ret = 0;
    while(right < n)
     {
     hash[s[right]]++; // 进⼊窗⼝
     while(hash[s[right]] > 1) // 判断
     hash[s[left++]]--; // 出窗⼝
     ret = max(ret, right - left + 1); // 更新结果
     right++; // 让下⼀个元素进⼊窗⼝
     }
     return ret;
     }
    };


http://www.kler.cn/a/377630.html

相关文章:

  • 关于大一上的总结
  • 3 抢红包系统
  • 基于Spring Boot的车辆违章信息管理系统(LW+源码+讲解)
  • leecode718.最长重复子数组
  • 【Cesium】三、实现开场动画效果
  • 30分钟学会LaTex
  • Google“Big Sleep“人工智能项目发现真实软件漏洞
  • 【算法赌场】区间合并
  • 传递悄悄话
  • java8有哪些新特性
  • 盘点谷歌全家桶中最值得付费的服务
  • 基于STL的自定义栈与队列实现:灵活选择底层容器的模板设计
  • 长度最小的子数组(滑动窗口)
  • cangjie仓颉程序设计-数据结构(四)
  • 无人机之远程指挥中心技术篇
  • 鸿蒙中的FA模型和Stage模型
  • 10.31.2024刷华为OD C题型
  • Scikit-learn和Keras简介
  • 为啥学习数据结构和算法
  • 最新整理:linux常见面试题库
  • 代码源NOIP DAY2 T1 LIS和LDS题解
  • Web Broker(Web服务应用程序)入门教程(5)
  • 2181、合并零之间的节点
  • PostgreSQL 删除重复数据
  • 【Eclipse系列】eclipse快捷键和设置
  • 群控系统服务端开发模式-应用开发-业务架构逻辑开发第一轮测试