双指针算法——滑动窗口

前言:

滑动窗口本质上也是利用双指针来解决特定情况下的问题。滑动窗口算法思想是通过俩个指针,定义在左边和右边,俩指针同向运动,保持着一个像“窗口”一样的双指针来不停的压缩或者扩展来移动“窗口”,从而找到特定的子数组。

滑动窗口基本做题思路:

首先我们可以利用暴力解法来看看优化,是否是利用双指针来解决?双指针是否同向移动?

如果满足,可以定义为使用滑动窗口来解决,时间复杂度可以大大优化(一般为0(N))

进窗口(right指针进入,相当于扩展窗口大小,框住子数组)

判断(判断题目给定条件是否成立,要是不符合该指定条件,则left指针选择移动)

出窗口(left指针移动,相当于缩小窗口大小,直到满足题目要求即可)

更新结果(更新题目所求,得出最优结果)

一、长度最小的子数组(. - 力扣(LeetCode)

如图所述,题目要求找出一连串的子数组,其数据之和要大于等于所给定的target,而且该子数组长度是最小的,以示例一举例:

算法思路:

我们可以利用双指针来不断寻找,定义在一个“窗口”里面来找他们元素之和是否大于等于给定条件的数,先“进窗口”,right指针不断插入,遍历一遍数组,然后再判定子数组里面的元素之和是否大于等于target,若是符合,则进入“出窗口”阶段,更新最小长度,将left、继续右移,判断是否符合;若是不符合,则继续由right扩展窗口:

代码实现:

二、最大连续1的个数 (. - 力扣(LeetCode)

如图所述,给定一个数组和目标数k,题目要求是最多翻转k个0,也就是说只要不超过k次,翻转多少都可以。

相当于就是在数组中把为0的数翻转成1,最多翻转k次,然后再择出最长的连续的1的子数组,并且返回该长度。

算法思路:

首先通过暴力解法可知道双指针是同向移动,从中得知可以使用滑动窗口算法,先定义一个左右指针,然后为了不改变原数组的任何数据,我们可以使用一个计数器来统计“窗口”里面的0个数:

代码实现:

三、将x减到0的最小操作数(. - 力扣(LeetCode)

如图所述,题目要求是每一次除去左端或右端数,除去的数相加若刚好为0,则返回除去的元素数量,若无法减为0,这返回-1。

算法思路:

对于该题,常规思路不好解,代码也会很复杂。我们可以逆向思维来解题:题目告知我们需要除去左端或右端元素来解决问题,但是我们可以从中间那一段来使用滑动窗口算法解决问题:

逆向思维想好思路后利用暴力解法,我们可以再次转化为我们熟悉的滑动窗口,只需从中间选出nums_sum - (a+b) 的元素之和即可,利用“窗口”来框住子数组,剩下的就是题目所需要的最终答案了,找出目标数可以参考第一题,基本逻辑是一样的:

代码实现:

四、找到字符串中所有字母异位词(. - 力扣(LeetCode)

如图所述,题目要求是返回p中的所有异位词,异位词的定义为:

这些都称为异位词,只需要将字符串中连续的子串异位词的开始索引记录返回即可

算法思路:

依旧是使用滑动窗口来解决,但是该题毕竟不是数组,也不能使用相加的方式来做。我们可以使用哈希表来完成该题。将p中的各个字符放入哈希表中记录下来,再定义一个用于统计s中符合的子串的哈希表,最终判断俩哈希表是否相同来返回起始索引:

代码实现:

注:由于题目要求只有26个字母,所以我们可以使用数组来代替哈希表,节省时间空间复杂度

public List<Integer> findAnagrams(String s, String p) {

        List<Integer> list = new ArrayList<>();

        int[] hash1 = new int[26];//让普通数组代替hash表,26个字母

        int[] hash2 = new int[26];

        int p_len = p.length();

        int len = 0;

        for (int i = 0; i < p_len; i++) hash1[p.charAt(i) - 97]++;//创建hash1表

        for (int left=0,right=0; right < s.length();right++){

            hash2[s.charAt(right) - 97] += 1;

            len++;//进窗口

            if(len > p_len){//判断

               hash2[s.charAt(left)-97]--;

               left++;//出窗口

               len--;

            }

            if(this.My_equals(hash1,hash2)) list.add(left);//更新结果

        }

        return list;

    }

    public  boolean My_equals(int[] hash1,int[] hash2) {

        for (int i = 0; i < hash2.length; i++)

            if (hash1[i] != hash2[i])

                return false;

        return true;

    }

五、最小覆盖子串(. - 力扣(LeetCode)

如图所述,只需要包含t中所有的字符内容都可以,不论包含几个,只需要大于等于t中的字符即可,最终返回最小的子串。

算法思路:

该题与上题类似,但也有不同。区别是上题只需要统计个数即可,但此题是只论种类,不论数目.

我们依旧是用到哈希表这个容器来帮助我们完成该题。为了防止像上题一样循环26次才得出俩哈希表是否相同,我们可以使用一个小小的优化来进行改造:

然后就可以使用我们的滑动窗口来解决该问题了:

代码实现:

public String minWindow(String s, String t) {

        int[] hash1 = new int[128];//hash来代替哈希表(26+26个字母)

        int[] hash2 = new int[128];

        char in = 0, out = 0;//用来统计出入窗口时的字符

        int hash1_size = 0, len = Integer.MAX_VALUE, flg = 0;//用于统计t的种类,和最短长度,起始位置

        for (char x : t.toCharArray()) {

            hash1[x] += 1;

            if (hash1[x] == 1) hash1_size++;//只有新种类进来时才加1

        }

        for (int left = 0, right = 0, count = 0/*维护hash2的种类*/; right < s.length(); right++) {

            in = s.charAt(right);

            hash2[in]++;//进窗口

            if (hash1[in] == hash2[in]) count++;//只有相等时才加1,防止重复种类的相加

            while (count == hash1_size) {//判断

                out = s.charAt(left);

                if (len > right - left + 1) {

                    len = right - left + 1;//更新长度

                    flg = left;//更新起始位置

                }

                if (hash1[out] == hash2[out]) count--;

                hash2[out]--;

                left++;//出窗口

            }

        }

        if(len == Integer.MAX_VALUE) return "";

        else return s.substring(flg,flg + len);

    }

以上就是关于滑动窗口的经典例题,希望对大家有所帮助,谢谢各位观看!

  • 73
    点赞
  • 65
    收藏
    觉得还不错? 一键收藏
  • 9
    评论
评论 9
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值