【练习】滑动窗口思想

这篇具有很好参考价值的文章主要介绍了【练习】滑动窗口思想。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【练习】滑动窗口思想,算法(Java),java,算法

  • 🎥 个人主页:Dikz12
  • 🔥个人专栏:Java算法
  • 📕格言:那些在暗处执拗生长的花,终有一日会馥郁传香
  • 欢迎大家👍点赞✍评论⭐收藏

目录

1.长度最小的子数组

题目解析 

讲解 

 代码实现

 2.无重复字符的最长子串

题目解析 

 讲解

代码实现 

3.最大连续1的个数 III

题目解析 

讲解 

代码实现 

4.将 x 减到 0 的最小操作数

题目解析

​编辑讲解 

代码实现 

 5.水果成篮

题目解析 

讲解 

代码实现 


1.长度最小的子数组

题目解析 

【练习】滑动窗口思想,算法(Java),java,算法

讲解 

解法一: 暴力枚举出所有子数组的和. O(n^2)

【练习】滑动窗口思想,算法(Java),java,算法 

解法二:利用“滑动窗口”优化.(利用单调性,同向双指针) O(n)

  1. left = 0,right = 0
  2. 进窗口
  3. 判断
  4.  出窗口 或者 是更新结果

【练习】滑动窗口思想,算法(Java),java,算法【练习】滑动窗口思想,算法(Java),java,算法

当sum 小于target时,就进窗口(right++);

满足判断条件(sum大于等于target)时,此时,更新结果,出窗口,可以⼤胆舍去 left得值然后left++,right也没有回退到left位置的必要了(left++)

 代码实现

    public int minSubArrayLen(int target, int[] nums) {
        int n = nums.length;
        int sum = 0, len = Integer.MAX_VALUE;
        for (int left = 0, right = 0; right < n; right++) {
            sum += nums[right];
            // 进窗口
            while (sum >= target) {
                // 更新len
                len = Math.min(len, (right - left + 1));
                // 出窗口
                sum -= nums[left++];
            }
        }
        return len == Integer.MAX_VALUE ? 0 : len;
    }

 2.无重复字符的最长子串

题目解析 

【练习】滑动窗口思想,算法(Java),java,算法

 讲解

解法一:暴力枚举 + 哈希表(判断字符是否重复出现)

枚举「从每⼀个位置」开始往后,⽆重复字符的⼦串可以到达什么位置。找出其中⻓度最⼤的即
可。在往后寻找⽆重复⼦串能到达的位置时,可以利⽤「哈希表」统计出字符出现的频次,来判断什么时候⼦串出现了重复元素.

解法二:使用“滑动窗口”来解决.

右端元素 ch 进⼊窗⼝的时候,使用数组模拟哈希来统计这个字符的频次:

  • 如果这个字符出现的频次超过1 ,说明窗⼝内有重复元素,那么就从左侧开始划出窗⼝,直到ch 这个元素的频次变为 1 ,然后再更新结果。
  • 如果没有超过1 ,说明当前窗⼝没有重复元素,可以直接更新结果
    【练习】滑动窗口思想,算法(Java),java,算法

代码实现 

    public int lengthOfLongestSubstring(String s) {
        //转换成字符数组
        char[] ch = s.toCharArray();
        int ret =0,n = s.length();
        //利用ascii码值模拟
        int[] hash = new int[128];
        for(int left = 0,right = 0; right < n; right++) {
            hash[ch[right]]++;
            while(hash[ch[right]] > 1) { //判断
                 hash[ch[left++]]--;//出窗口
            }
            ret = Math.max(ret,right - left + 1);
        }
        return ret;

3.最大连续1的个数 III

题目解析 

【练习】滑动窗口思想,算法(Java),java,算法

讲解 

 意思就是:找出最长的子数组,且 0 的个数不能超过 k.

解法一: 暴力枚举 + 计数器(统计0出现的次数)

解法二:滑动窗口

  1. 定义left=0,right=0 ; count=0 统计0的次数;
  2.  进窗口,如果 等于 0,count++; 等于 1,不做处理
  3. 判断,当 count 大于 k,出窗口,出的时候,等于 0,count - -;等于 1,不做处理
  4. 更新结果. (不满足判断,直接更新结果;否则,先出窗口,在更新)

【练习】滑动窗口思想,算法(Java),java,算法 【练习】滑动窗口思想,算法(Java),java,算法

代码实现 

    public int longestOnes(int[] nums, int k) {
        int n = nums.length, count = 0, ret = 0;
        for (int left = 0, right = 0; right < n; right++) {
            if (nums[right] == 0) {
                count++;
            }
            while (count > k) { // 判断
                if (nums[left] == 0) {
                    count--;
                }
                left++; // 出窗口
            }
            ret = Math.max(ret, right - left + 1); // 更新结果
        }
        return ret;
    }

4.将 x 减到 0 的最小操作数

题目解析

讲解 

正难则反 :题⽬要求的是数组「左端+右端」两段连续的、和为 x 的最短数组. 转换成求数组内⼀段连续的、和为 sum(nums) - x 的最⻓数组.

【练习】滑动窗口思想,算法(Java),java,算法

 解法:滑动窗口. 

这是的写法就和第一题的时候就很相似了.(求等于target的最长的子数组) 

  1.  计算出 target = sum - x的值,target 小于 0,直接返回 -1;(sum为整个数组的和)
  2. 进窗口  right++(记录下当前和tmp)
  3. 判断. 当tmp大于target时,出窗口,先从tmp中舍去当前位置的值,在left++;
  4. 更新结果. 我们要的是 等于target 的子数组长度,需要加上个if条件.

代码实现 

    public int minOperations(int[] nums, int x) {
        int ret = -1,sum = 0,n = nums.length,tmp = 0;
        for(int i : nums) {
            sum += i;
        }
        int target = sum - x;
        if(target < 0) {
            return -1;
        }
        for(int left = 0,right = 0; right < n;right++) {
            //进窗口
            tmp += nums[right]; 
            //判断
            while(tmp > target) { 
              //出窗口
              tmp -= nums[left++];
            }
            if(tmp == target) {
                ret = Math.max(ret,right - left + 1);
            }
        }
        if(ret == -1) {
            return -1;
        }else{
            //ret是sum-x的长度
            return n - ret;
        }
    }

 5.水果成篮

题目解析 

【练习】滑动窗口思想,算法(Java),java,算法

 这么长的小作文,意思就是:找出一个最长的子数组的长度,子数组中不超过两种类型的水果.

讲解 

解法一:暴力枚举 + Map 容器. 

解法二:滑动窗口 .

【练习】滑动窗口思想,算法(Java),java,算法

 

指针同向移动,可以使用“滑动窗口.

  1.  初始化Map容器(也可以使用数组模拟),来统计窗⼝内⽔果的种类和数量;
  2. 初始化变量:左右指针 left = 0,right = 0,记录结果的变量 ret = 0
  3. 进窗口.将当前⽔果放⼊Map中;
  4. 判断.当前⽔果进来后,Map的⼤⼩:如果超过 2:将左侧元素滑出窗⼝,并且在Map中将该元素的value减⼀;如果这个元素的value减⼀之后变成了 0,就把该元素从哈希表中删除;
    重复上述两个过程,直到哈希表中的⼤⼩不超过 2;
  5.  更新结果

【练习】滑动窗口思想,算法(Java),java,算法

代码实现 

    public int totalFruit(int[] fruits) {
        int n = fruits.length, ret = 0;
        Map<Integer, Integer> map = new HashMap<>();
        for (int left = 0, right = 0; right < n; right++) {
            int key = fruits[right];
            map.put(key, map.getOrDefault(key, 0) + 1);// 进窗口
            // 判断
            while (map.size() > 2) {
                // 出窗口
                int out = fruits[left];
                map.put(out, map.get(out) - 1);
                if (map.get(out) == 0) {
                    map.remove(out);
                }
                left++;
            }
            ret = Math.max(ret, right - left + 1);
        }
        return ret;
    }

 数组模拟实现:文章来源地址https://www.toymoban.com/news/detail-844636.html

   public int totalFruit(int[] fruits) {
        int n = fruits.length, ret = 0;
        int[] map = new int[n+1];
        // Map<Integer, Integer> map = new HashMap<>();
        for (int left = 0, right = 0,type = 0; right < n; right++) {
            int key = fruits[right];
            // map.put(key, map.getOrDefault(key, 0) + 1);// 进窗口
            if(map[key] == 0) {
                type++; //记录水果种类
            }
            map[key]++; //进窗口
            // 判断
            while (type > 2) {
                // 出窗口
                int out = fruits[left];
                map[out]--;
                // map.put(out, map.get(out) - 1);
                if (map[out] == 0) {
                    type--;
                }
                left++;
            }
            ret = Math.max(ret, right - left + 1);
        }
        return ret;
    }

到了这里,关于【练习】滑动窗口思想的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • leetcode 567. 字符串的排列(滑动窗口-java)

    难度 -中等 leetcode567. 字符串的排列 给你两个字符串 s1 和 s2 ,写一个函数来判断 s2 是否包含 s1 的排列。如果是,返回 true ;否则,返回 false 。 换句话说,s1 的排列之一是 s2 的 子串 。 示例 1: 输入:s1 = “ab” s2 = “eidbaooo” 输出:true 解释:s2 包含 s1 的排列之一 (“ba”

    2024年02月10日
    浏览(32)
  • 【剑指offer刷题记录 java版】数组双指针 之 滑动窗口

    本系列文章记录labuladong的算法小抄中剑指offer题目 题目链接:https://leetcode.cn/problems/zui-chang-bu-han-zhong-fu-zi-fu-de-zi-zi-fu-chuan-lcof/ 题目链接:https://leetcode.cn/problems/MPnaiL/ 题目链接:https://leetcode.cn/problems/VabMRr/ 题目链接:https://leetcode.cn/problems/wtcaE1/ 题目链接:https://leetcode.cn/pr

    2024年02月09日
    浏览(43)
  • Java【手撕滑动窗口】LeetCode 209. “长度最小子数组“, 图文详解思路分析 + 代码

    各位读者好, 我是小陈, 这是我的个人主页, 希望我的专栏能够帮助到你: 📕 JavaSE基础: 基础语法, 类和对象, 封装继承多态, 接口, 综合小练习图书管理系统等 📗 Java数据结构: 顺序表, 链表, 堆, 二叉树, 二叉搜索树, 哈希表等 📘 JavaEE初阶: 多线程, 网络编程, TCP/IP协议, HTTP协议

    2024年02月11日
    浏览(36)
  • 【免费题库】华为OD机试 - 滑动窗口最大和(Java & JS & Python & C & C++)

    哈喽,本题库完全免费,收费是为了防止被爬,大家订阅专栏后可以私信联系退款。感谢支持 有一个N个整数的数组,和一个长度为M的窗口,窗口从数组内的第一个数开始滑动直到窗口不能滑动为止, 每次窗口滑动产生一个窗口和(窗口内所有数的和),求窗口滑动产生的所

    2024年04月10日
    浏览(37)
  • Java【手撕滑动窗口】LeetCode 3. “无重复字符的最长子串“, 图文详解思路分析 + 代码

    各位读者好, 我是小陈, 这是我的个人主页, 希望我的专栏能够帮助到你: 📕 JavaSE基础: 基础语法, 类和对象, 封装继承多态, 接口, 综合小练习图书管理系统等 📗 Java数据结构: 顺序表, 链表, 堆, 二叉树, 二叉搜索树, 哈希表等 📘 JavaEE初阶: 多线程, 网络编程, TCP/IP协议, HTTP协议

    2024年02月11日
    浏览(29)
  • Java【手撕滑动窗口】LeetCode 438. “字符串中所有异位词“, 图文详解思路分析 + 代码

    各位读者好, 我是小陈, 这是我的个人主页, 希望我的专栏能够帮助到你: 📕 JavaSE基础: 基础语法, 类和对象, 封装继承多态, 接口, 综合小练习图书管理系统等 📗 Java数据结构: 顺序表, 链表, 堆, 二叉树, 二叉搜索树, 哈希表等 📘 JavaEE初阶: 多线程, 网络编程, TCP/IP协议, HTTP协议

    2024年02月10日
    浏览(31)
  • 【华为OD机试真题 C++ Java Python】1、滑动窗口最大值 | 机试真题+思路参考+代码解析

    🍂个人博客首页: KJ.JK   🍂专栏介绍: 定期更新华为OD各个时间阶段的机试真题,每日定时更新,本专栏每篇的文章都会将使用C++、Python、Java三种语言进行更新解答,每个题目的思路分析都非常详细,超过百字欢迎大家订阅学习,代码可以直接运行使用 🎃题目描述 有一个

    2024年02月08日
    浏览(33)
  • 拓扑排序 (算法思想+图解+模板+练习题)

    有向无环图一定是拓扑序列,有向有环图一定不是拓扑序列。 无向图没有拓扑序列。 首先我们先来解释一下什么是有向无环图: 有向就是我们两个结点之间的边是有方向的,无环的意思就是整个序列中没有几个结点通过边形成一个圆环。 下图就是一个有向无环图,它也一定

    2024年02月03日
    浏览(36)
  • LeetCode算法小抄--滑动窗口算法

    ⚠申明: 未经许可,禁止以任何形式转载,若要引用,请标注链接地址。 全文共计6244字,阅读大概需要3分钟 🌈更多学习内容, 欢迎👏关注👀文末我的个人微信公众号:不懂开发的程序猿 个人网站:https://jerry-jy.co/ 滑动窗口算法 思路 1、我们在字符串 S 中使用双指针中的

    2023年04月09日
    浏览(31)
  • 【算法】基础算法002之滑动窗口(一)

    👀 樊梓慕: 个人主页  🎥 个人专栏: 《C语言》 《数据结构》 《蓝桥杯试题》 《LeetCode刷题笔记》 《实训项目》 《C++》 《Linux》《算法》 🌝 每一个不曾起舞的日子,都是对生命的辜负 目录 前言 1.长度最小的子数组 滑动窗口类问题解题思路大纲: 2.无重复字符的最长

    2024年02月19日
    浏览(35)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包