【LeetCode热题100】打卡第42天:滑动窗口最大值搜索二维矩阵II

文章目录

  • 【LeetCode热题100】打卡第42天:滑动窗口最大值&搜索二维矩阵II
    • ⛅前言
  • 滑动窗口最大值
    • 🔒题目
    • 🔑题解
  • 搜索二维矩阵II
    • 🔒题目
    • 🔑题解

【LeetCode热题100】打卡第42天:滑动窗口最大值&搜索二维矩阵II

⛅前言

大家好,我是知识汲取者,欢迎来到我的LeetCode热题100刷题专栏!

精选 100 道力扣(LeetCode)上最热门的题目,适合初识算法与数据结构的新手和想要在短时间内高效提升的人,熟练掌握这 100 道题,你就已经具备了在代码世界通行的基本能力。在此专栏中,我们将会涵盖各种类型的算法题目,包括但不限于数组、链表、树、字典树、图、排序、搜索、动态规划等等,并会提供详细的解题思路以及Java代码实现。如果你也想刷题,不断提升自己,就请加入我们吧!QQ群号:827302436。我们共同监督打卡,一起学习,一起进步。
博客主页💖:知识汲取者的博客
LeetCode热题100专栏🚀:LeetCode热题100

Gitee地址📁:知识汲取者 (aghp) - Gitee.com

题目来源📢:LeetCode 热题 100 - 学习计划 - 力扣(LeetCode)全球极客挚爱的技术成长平台

PS:作者水平有限,如有错误或描述不当的地方,恳请及时告诉作者,作者将不胜感激

滑动窗口最大值

🔒题目

原题链接:239.滑动窗口最大值

image-20230721101215009

🔑题解

  • 解法一:递减队列

    所谓的递减队列就是队头到队尾的元素按照从大到小的顺序进行排序的

    算法的核心步骤:

    1. 添加元素,要保障单调队列中的元素是递减的,由于添加元素是在队尾进行的,所以要求队列中其他元素都要比它大
    2. 移除元素,由于移除元素是在队头进行的,如果当前元素是最大值,需要移除单调队列的队头元素(队头元素就是当前最大值)

    对于这种题目我现在也不是一开始就没有思路了,一下就想到肯定要使用 递减队列 ,但是目前实现起来还是有一点困难,经过不断的debug调试,然后再不断优化代码的语句和逻辑判断,最终还是实现了,现在我就来说说我实现的思路吧:我们①首先需要构建两个队列,一个队列(可以是普通队列)用于存储滑动窗口中的元素,一个队列(只能是双端队列)用于记录当前窗口最大值,②这个双端队中的元素是递减的,这样能够保障队尾元素总是当前最大元素,我们要获取当前最大元素,只需要获取队尾元素即可③移除元素时,如果发现移除的元素是当前最大值,需要移除双端队列的队尾元素,这里不罗嗦了,直接画图吧

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-m8f6nOFf-1690024629049)(https://gitee.com/aghp/typora-img/raw/master/csdn/202307221916698.gif)]

    本题可以参考:【LeetCode热题100】打卡第35天:最小栈

    import java.util.Deque;
    import java.util.LinkedList;
    
    /**
     * @author ghp
     * @title
     */
    public class Solution {
        public int[] maxSlidingWindow(int[] nums, int k) {
            int[] ans = new int[nums.length - k + 1];
            DescQueue queue = new DescQueue();
            // 初始化窗口
            for (int i = 0; i < k; i++) {
                queue.add(nums[i]);
            }
            int j = 0;
            ans[j++] = queue.getMax();
            // 向右滑动窗口
            for (int i = k; i < nums.length; i++) {
                queue.remove();
                queue.add(nums[i]);
                ans[j++] = queue.getMax();
            }
            return ans;
        }
    }
    
    class DescQueue {
        private Deque<Integer> queue;
        private Deque<Integer> maxQueue;
    
        public DescQueue() {
            this.queue = new LinkedList<>();
            this.maxQueue = new LinkedList<>();
        }
    
        /**
         * 新增元素到队列尾部
         *
         * @param val
         */
        public void add(int val) {
            queue.offer(val);
            while (!maxQueue.isEmpty() && maxQueue.peekLast() < val) {
                // 新增元素大于maxQueue队尾元素,不符合递减队列的要求
                // 需要移除队尾元素,直到maxQueue符合递减队列为止
                maxQueue.pollLast();
            }
            // 新增元素大于等于队尾元素,符合递增队列,直接添加到队尾
            maxQueue.offer(val);
        }
    
        /**
         * 移除队列头部元素
         */
        public void remove() {
            int val = queue.poll();
            if (val == maxQueue.peek()) {
                // 移除的队头元素是当前最大值,则需要更新maxQueue
                maxQueue.poll();
            }
        }
    
        /**
         * 获取当前队列中最大元素
         *
         * @return
         */
        public int getMax() {
            // maxQueue的头部元素即为当前最大值,直接返回即可
            return maxQueue.peek();
        }
    }
    

    复杂度分析:

    • 时间复杂度: O ( n ) O(n) O(n)
    • 空间复杂度: O ( n ) O(n) O(n)

    其中 n n n 为数组中元素的个数

    这里需要了解一下队列的几个常用API,不要搞混了,队列是队尾进队头出,队尾操作有:offerpollLastpeekLastgetLast;队头操作有:pollpeekelementgetFirst

    代码优化:空间优化

    前面我们使用了两个队列,一个队列存储窗口中的元素,一个队列记录当前最大值,我们可以发现对于本题,我们其实可以发现我们只需要知道当前窗口中的最大值即可,并不需要去记录当前窗口中所有的元素,所以这里我们可以对上面的代码进行一个改造,从而省略一个队列的开销,从而一定程度降低空间复杂度,那么我们该如何实现呢,这里给出主要思路:

    1. 添加元素,要保障单调队列中的元素是递减的,由于添加元素是在队尾进行的,所以要求队列中其他元素都要比它大
    2. 移除元素,我们不必像之前一样直接移除左边界元素,但是我们队列中存储的数组的索引,我们可以换一种思路,只有当最大值被移除时,才更新递减队列(DescQueue)

    总的思路还是和上一题大差不差的,空间复杂度没有降低,但是减少了内存占用

    备注:优化参考LeetCode官方题解

    import java.util.Deque;
    import java.util.LinkedList;
    
    /**
     * @author ghp
     * @title
     */
    public class Solution {
        public int[] maxSlidingWindow(int[] nums, int k) {
            int[] ans = new int[nums.length - k + 1];
            DescQueue queue = new DescQueue();
            // 初始化窗口
            for (int i = 0; i < k; i++) {
                queue.add(nums, i);
            }
            ans[0] = nums[queue.getMax()];
            // 向右滑动窗口
            for (int i = k; i < nums.length; i++) {
                queue.remove(i - k);
                queue.add(nums, i);
                ans[i - k + 1] = nums[queue.getMax()];
            }
            return ans;
        }
    }
    
    class DescQueue {
        private Deque<Integer> maxQueue;
    
        public DescQueue() {
            this.maxQueue = new LinkedList<>();
        }
    
        /**
         * 新增元素到队列尾部
         *
         * @param arr
         * @param i
         */
        public void add(int[] arr, int i) {
            while (!maxQueue.isEmpty() && arr[i] > arr[maxQueue.peekLast()]) {
                // 新增元素大于maxQueue队尾元素,不符合递减队列的要求
                // 需要移除队尾元素,直到maxQueue符合递减队列为止
                maxQueue.pollLast();
            }
            // 新增元素大于等于队尾元素,符合递增队列,直接添加到队尾
            maxQueue.offer(i);
        }
    
        /**
         * 移除队列头部元素
         */
        public void remove(int i) {
            while (!maxQueue.isEmpty() && maxQueue.peek() <= i) {
                // 队头元素的索引已经超出了左边界,直接移除(队头元素已过期)
                maxQueue.poll();
            }
        }
    
        /**
         * 获取当前队列中最大元素
         *
         * @return
         */
        public int getMax() {
            // maxQueue的头部元素即为当前最大值,直接返回即可
            return maxQueue.peek();
        }
    }
    
  • 解法二:优先队列

    优先队列可能比前面的单调队列还要简单很多,思路也超级简单,这里就简单讲解一下:

    1. 构造优先队列(这一步是核心步骤),我们创建一个优先队列,队列中的元素是一个数组,数组的第一个元素是添加的nums数组的值,第二个元素是nums数组值对应的索引
    2. 添加元素,直接添加即可,优先队列底层会通过维护一个大根堆实现降序排序,即队头元素到队尾元素的值从大到小排序

    其实说白了优先队列本质也是一个单调队列思想,只是这个单调对了不需要我们自己维护,相较于解法一,本方法就简单太多了

    import java.util.PriorityQueue;
    
    /**
     * @author ghp
     * @title
     */
    public class Solution {
        public int[] maxSlidingWindow(int[] nums, int k) {
            int[] ans = new int[nums.length - k + 1];
            // 创建一个优先队列,并且根据数组第一个元素进行降序排列,第一个元素是值,第二个元素是索引
            PriorityQueue<int[]> pq = new PriorityQueue<>((a, b) -> b[0] - a[0]);
            // 初始化窗口
            for (int i = 0; i < k; i++) {
                pq.offer(new int[]{nums[i], i});
            }
            int j = 0;
            ans[j++] = pq.peek()[0];
            // 向右滑动窗口
            for (int i = k; i < nums.length; i++) {
                while (!pq.isEmpty() && pq.peek()[1] <= i - k) {
                    // 队头元素的索引已经超出了左边界,直接移除(队头元素已过期)
                    pq.poll();
                }
                pq.offer(new int[]{nums[i], i});
                ans[j++] = pq.peek()[0];
            }
    
            return ans;
        }
    }
    

    复杂度分析:

    • 时间复杂度: O ( n ) O(n) O(n)
    • 空间复杂度: O ( n l o g k ) O(nlogk) O(nlogk)

    其中 n n n 为数组中元素的个数, k k k是窗口的大小

搜索二维矩阵II

🔒题目

原题链接:240.搜索二维矩阵II

image-20230722181524442

🔑题解

  • 解法一:暴力枚举(能过)

    class Solution {
        public boolean searchMatrix(int[][] matrix, int target) {
            for (int[] row : matrix) {
                for (int element : row) {
                    if (element == target) {
                        return true;
                    }
                }
            }
            return false;
        }
    }
    

    复杂度分析

    • 时间复杂度: O ( m n ) O(mn) O(mn)
    • 空间复杂度: O ( 1 ) O(1) O(1)

    其中m是行,n是列

  • 解法二:二分查找

    二分查找的简单应用,这里就不展开讲了,唯一需要注意的就是边界问题,对于边界问题的判断,实在不会判断的,有两种选择,一种是直接举例列出边界条件,一种是记口诀,像这种经典的算法,一般都有前辈总结的口诀,但是我还是比较推荐使用距离列出边界条件,记忆口诀这个东西太死了,失去了算法本身的意义,算法不是为了死记硬背,而是在于理解

    此外和这个题目相似的题还有很多,可以直接搜索关键词:二分查找

    /**
     * @author ghp
     * @title
     */
    class Solution {
        public boolean searchMatrix(int[][] matrix, int target) {
            int row = matrix.length;
            for (int i = 0; i < row; i++) {
                Integer result = binarySearch(matrix[i], target);
                if (result != null) {
                    return true;
                }
            }
            return false;
        }
    
        private Integer binarySearch(int[] arr, int target) {
            int l = 0, r = arr.length-1;
            while (l <= r) {
                int mid = (r - l) / 2 + l;
                if (target == arr[mid]) {
                    return mid;
                } else if (target < arr[mid]) {
                    r = mid - 1;
                } else {
                    l = mid + 1;
                }
            }
            return null;
        }
    }
    

    复杂度分析:

    • 时间复杂度: O ( m l o g n ) O(mlogn) O(mlogn)
    • 空间复杂度: O ( 1 ) O(1) O(1)

    其中 n n n 为数组中元素的个数

  • 解法三:Z字形查找

    这个解法来自LeetCode官网,真TM强,没想到啊啊啊啊😫

    核心思路:充分利用层与层之间的排序关系,从右上角开始,不断比较当前坐标的值和目标值,缩小搜索范围,每一次比较都能缩小一行或一列的数据

    PS:能画一个动图就好了,我相信大家一看就懂(大家脑补一下吧,画图太费时间了),这个解法理解起来并不难,就是想不到啊

    class Solution {
        public boolean searchMatrix(int[][] matrix, int target) {
            int m = matrix.length, n = matrix[0].length;
            int row = 0, col = n - 1;
            while (row < m && col >= 0) {
                if (matrix[row][col] == target) {
                    return true;
                }
                if (matrix[row][col] > target) {
                    // 目标值比当前坐标值要小,则向左逼近
                    --col;
                } else {
                    // 目标值比当前坐标值要大,则向下逼近
                    ++row;
                }
            }
            return false;
        }
    }
    

    复杂度分析

    • 时间复杂度: O ( m + n ) O(m+n) O(m+n),每次都会缩小一行或一列的数据,所以最坏情况,时间复杂度是 O ( m + n ) O(m+n) O(m+n),也就是目标值在左下角
    • 空间复杂度: O ( 1 ) O(1) O(1)

    其中m是行,n是列

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/43275.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

帖子列表和SerializerMixin注意事项

帖子序列化 继承SerializerMixin 即可调用to_dict()序列化 后端 class PostModel(db.Model, SerializerMixin):serialize_only ("id", "title", "content", "create_time", "board", "author")__tablename__ …

rabbitmq访问异常

看到这个问题&#xff0c;第一时间想到rabbitmq的问题&#xff0c;应该权限导致的 先创建virtual hosts 接着创建用户并赋予权限,将eayc的virtual hosts权限赋予acc用户即可 15:34:24.250 WARN com.rabbitmq.client.impl.ForgivingExceptionHandler - An unexpected connec…

线程的基本概念

线程的基本概念 1. 概念1.1 什么是线程1.2 为什么要有线程1.3 进程和线程的区别 2. 线程创建的基本方法3. Thread 类3.1. Thread 的常见构造方法3.2 Thread 类常见的几种方法 4. 线程的状态 1. 概念 1.1 什么是线程 一个线程就是一个 “执行流”. 每个线程之间都可以按照顺讯…

Vue中TodoList案例_添加

与上一篇Vue中TodoList案例_初始化列表有四个文件变化了 安装nanoid库&#xff1a; npm i nanoid App.vue <template><div id"root"><div class"todo-container"><div class"todo-wrap"><MyHeader :addTodo"…

IDEA快速创建SpringBoot

文件具有错误的版本 61.0, 应为 52.0报错可以看看是不是Springboot的版本比较高 和jdk版本不匹配 package com.qf.controller;import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframewor…

【Java】 服务器cpu过高如何排查和解决?

文章目录 前言一、常见能够引起CPU100%异常的情况都有哪些&#xff1f;二、服务器CPU使用率飙升异常&#xff0c;黄金4步排查法三、排查 CPU 故障的常用命令四、什么场景会造成 CPU 低而负载确很高呢&#xff1f;五、监控发现线上机器内存占用率居高不下&#xff0c;如何分析进…

第一天基础名词

文章目录 一、域名1、域名的概念2、域名注册3、域名的分类 二、DNS1、DNS的概念2、DNS解析3、本地hosts文件与DNS的关系4、如何查看本地Hosts文件 三、CDN1、CDN的概念2、CDN原理&#xff08;1&#xff09;回顾域名解析&#xff08;2&#xff09;CDN原理 3、常见DNS攻击 四、脚…

HideSeeker论文阅读

文章目录 3.1 Overview of Our System HideSeeker3.2 Visual Information Extraction3.3 Relation Graph Learning3.4 Hidden Object Inference 4 EVALUATIONS4.7 Summary 6 DISCUSSIONS AND CONCLUSION 3.1 Overview of Our System HideSeeker 我们设计了一种名为“HideSeeke…

【ARM Cache 系列文章 2 -- Cache Coherence及内存顺序模学习】

文章目录 Cache Coherence 背景1.1 内存顺序模型简介(Memory Model)1.1.1 Normal Memory1.1.2 Device Memory 1.2 Cache 一致性问题解决方案1.2.1 Shareability 属性1.2.2 Non-Shareable 属性1.2.3 Inner-Shareable 属性1.2.4 Out-Shareable 属性 1.3 Shareability 和 PoC/PoU …

Python Flask构建微信小程序订餐系统 (十一)

🔥 已经删除的会员不允许进行编辑昵称 🔥 🔥 已经删除的会员要隐藏掉会员信息的编辑按钮 🔥 🔥 创建商品表 food 🔥 CREATE TABLE `food` (`id` int(11) unsigned NOT NULL AUTO_INCREMENT,`cat_id` int(11) NOT NULL DEFAULT 0 COMMENT 分类id,`name` varchar…

# **基于TiDB Binlog架构的主备集群切换操作手册**

作者&#xff1a; Liuhaoao 原文来源&#xff1a; https://tidb.net/blog/dc65ef62 操作背景&#xff1a;最近手头有个系统&#xff0c;刚做完灾备建设及数据同步&#xff08; 文章链接在这 &#xff09;&#xff0c;需要进行灾备切换演练&#xff0c;验证灾备库建设是否…

【IDEA】IDEA 中点击Download Source(下载源码)后,下载的源码存储到哪了?

【问】如上图&#xff0c;IDEA 中点击 Download Source&#xff08;下载源码&#xff09;后&#xff0c;下载的源码存储到哪了&#xff1f; 【答】 先找到此源码所属 Jar 包在哪&#xff1b;点击 Download Source&#xff08;下载源码&#xff09;后&#xff0c;会发现存储 J…

Android Studio Flutter 开发配置

近来比较闲&#xff0c;就研究下Flutter 开发&#xff0c;在此记录下studio 配置过程&#xff0c;时间是2023.7.19 在 Windows 操作系统上安装和配置 Flutter 开发环境 1.首先下载 Flutter SDKhttps://storage.flutter-io.cn/flutter_infra_release/releases/stable/windows/…

静态html引入ucharts并直接使用组件标签

由于官方不能直接使用qiun-vue-ucharts在静态html页面使用。 DIY可视化对此类库进行了改进&#xff0c;把它的包独立打包成一个可以依赖的JS。 首先定义一个核心JS&#xff0c;用于打包生成uchart import qiunVueUcharts from qiun/vue-ucharts;const install (app) > {…

【学习笔记】Unity基础(十)【Unity界面之Scene窗口工具栏】

目录 一 总览二 Scene 视图导航2.1 Orientation 场景视角2.2 场景视角的移动、旋转和缩放2.3 场景对象聚焦 三 对象可见性与场景可见性3.1 对象可见性控件3.2 场景可见性控件3.3 Isolate selected GameObjects 四 Overlays五 Position GameObjects5.1 Transform Tools5.2 Tool …

设计模式与技术场景面试题详解

工厂方法模式 简单工厂模式 工厂方法模式 抽象工厂模式 策略模式 工厂模式策略模式&#xff08;登录案例&#xff09; 责任链模式 技术场景 单点登录怎么实现 权限认证是如何实现的 上传数据的安全性你们怎么控制 你负责的项目遇到了哪些棘手的问题&#xff1f;怎么解决的 你们…

部署ELK+Kafka+Filebeat日志收集分析系统

部署ELKKafkaFilebeat日志收集分析系统 文章目录 部署ELKKafkaFilebeat日志收集分析系统一、ELK 简介1、ELK日志分析系统组成2、Elasticsearch&#xff08;es&#xff09;3、Logstash4、Kibana5、日志处理步骤 二、Elasticsearch介绍1、Elasticsearch核心概念2、开启分片副本的…

用户管理功能

后端 # 获取用户列表 bp.get("/user/list") def user_list():users UserModel.query.order_by(UserModel.join_time.desc()).all()user_list [user.to_dict() for user in users]return restful.ok(datauser_list)# 用户是否可用 bp.post("/user/active"…

kettle开发-Day40-AI分流之case/switch

前言&#xff1a; 前面我们讲到了很多关于数据流的AI方面的介绍&#xff0c;包括自定义组件和算力提升这块的&#xff0c;今天我们来学习一个关于kettle数据分流处理非常重要的组件Switch / Case 。当我们的数据来源于类似日志、csv文件等半结构化数据时&#xff0c;我们需要在…

Java多线程基础复习

文章目录 多线程1.进程进程属性并发和并行虚拟地址空间 2.线程概念线程的创建方式 3.Thread类常见构造方法和属性线程的状态优先级后台线程线程是否存活start和run 4. 线程的一些基本操作线程中断(interrupted)线程等待joincurrentThread&#xff08;获取当前线程引用&#xff…