【LeetCode力扣】单调栈解决Next Greater Number(下一个更大值)问题

目录

1、题目介绍

2、解题思路

2.1、暴力破解法

2.2、经典Next Greater Number问题解法


1、题目介绍

原题链接:496. 下一个更大元素 I - 力扣(LeetCode)

示例1:

输入:nums1 = [4,1,2], nums2 = [1,3,4,2].
输出:[-1,3,-1]
解释:nums1 中每个值的下一个更大元素如下所述:
- 4 ,用加粗斜体标识,nums2 = [1,3,4,2]。不存在下一个更大元素,所以答案是 -1 。
- 1 ,用加粗斜体标识,nums2 = [1,3,4,2]。下一个更大元素是 3 。
- 2 ,用加粗斜体标识,nums2 = [1,3,4,2]。不存在下一个更大元素,所以答案是 -1 。

实例2:

输入:nums1 = [2,4], nums2 = [1,2,3,4].
输出:[3,-1]
解释:nums1 中每个值的下一个更大元素如下所述:
- 2 ,用加粗斜体标识,nums2 = [1,2,3,4]。下一个更大元素是 3 。
- 4 ,用加粗斜体标识,nums2 = [1,2,3,4]。不存在下一个更大元素,所以答案是 -1 。

提示:

  • 1 <= nums1.length <= nums2.length <= 1000
  • 0 <= nums1[i], nums2[i] <= 10^4
  • nums1和nums2中所有整数 互不相同
  • nums1 中的所有整数同样出现在 nums2 中

2、解题思路

        该题如果使用暴力破解法,其代码实现过程是很容易想到的,只需要找到nums1此时的元素在nums2中的位置,并向右查询是否存在更大的值,有则返回该值,无则返回-1即可。结合思想不难想到时间复杂度为O(m*n),m和n分别为两个数组各自的长度。虽然该题直接使用暴力破解法可以直接通过,但是只是出题人没有为难大家,如果该题的数据非常庞大,此时直接使用暴力破解法必然会导致超时。而本文将讲解单调队列的算法模版解决这类问题,它能够很好的将时间复杂度控制在O(m+n)。

下面将使用两种方法来解题,一种是暴力破解法,一种是Next Greater Number问题解法。

2.1、暴力破解法

从头开始遍历nums1,每次遍历从nums2中找到对应元素,然后从该元素的下一个元素开始依次比较,找出大于该值的第一个元素即可。

【代码实现】 

class Solution {public int[] nextGreaterElement(int[] nums1, int[] nums2) {int m = nums1.length, n = nums2.length;int[] res = new int[m];for (int i = 0; i < m; ++i) {int j = 0;while (j < n && nums2[j] != nums1[i]) {  //在nums2中找到该值++j;}int k = j + 1;while (k < n && nums2[k] < nums2[j]) {   //从该值的下一个元素开始向右查找++k;}res[i] = k < n ? nums2[k] : -1;    //超出nums2数组长度则表示不存在比该值大的数,返回-1}return res;}
}

时间复杂度:O(m*n),m和n分别为两数组的长度。

空间复杂度:O(1)。

2.2、经典Next Greater Number问题解法

首先需要补充一些知识点,下面用一个比较抽象的例子讲解:

把数组的元素想象成并列站立的人,元素大小想象成人的身高。视线从左往右,这些人面对你站成一列。如何求第一个元素【2】的下一个更大值呢?,其实很简单,只要没被该元素【2】挡住的第一个元素,就是【2】的下一个更大值,如下图所示,第一个【2】没有挡住的第一个元素就是【4】,此时【4】就是【2】的下一个更大值。

当理解了这个前提后,我们从后往前遍历该数组:

【3】,【3】背后看不到任何元素,即没有比它更大的右元素,因此next greater为-1;

【4】,【4】背后依然看不到任何元素(3被挡住了),此时next greater为-1;

【2】,【2】背后看到的第一个元素是4,因此next greater为4;(绿色2)

【1】,【1】背后看到的第一个元素是2,因此next greater为2;

【2】,【2】背后看到的第一个元素是4,因此next greater为4;(蓝色2)

在代码中为了记录某个值背后没被挡住的第一个元素,这里使用栈stack来记录,入栈出栈规则:当前值比栈中元素大时(证明被挡住了),将该值出栈,直至栈空或当前值小于栈顶元素,此时栈顶元素即为next greater;随后将当前值入栈。

因此我们需要做的就是从后往前遍历nums2数组,并计算出nums2对于的next greater,同时使用Map将【nums2的该值】以及【该值的next greater】记录下来,最后遍历nums1数组,从Map中取出对应值的next greater存入返回数组ret中即可。下面是代码实现部分。

【代码实现】 

class Solution {public int[] nextGreaterElement(int[] nums1, int[] nums2) {Stack<Integer> stack = new Stack<>();Map<Integer,Integer> map = new HashMap<>();for(int i = nums2.length - 1; i >= 0; i--) {   //从后往前int num = nums2[i];while(!stack.isEmpty() && stack.peek() <= num) {   //栈内元素小于等于当前值时,出栈stack.pop();       //让栈始终保持只有【大于】当前值的元素}map.put(num,stack.isEmpty() ? -1 : stack.peek());   //map记录该值的下一个最大值,没有则-1stack.push(num);}int[] ret = new int[nums1.length];for(int i = 0; i < nums1.length; i++) {    //将map中的对应值取出存入返回数组ret[i] = map.get(nums1[i]);}return ret;}
}

时间复杂度:O(m+n),m和n分别为两数组的长度。

空间复杂度:O(n),用于存储哈希表。

更多【LeetCode刷题】推荐:

【LeetCode力扣】面试题 17.14. 最小K个数(top-k问题)-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/zzzzzhxxx/article/details/135737266?spm=1001.2014.3001.5501 【LeetCode力扣】42. 接雨水-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/zzzzzhxxx/article/details/134104222?spm=1001.2014.3001.5501

 【LeetCode力扣】189 53 轮转数组 | 最大子数组和-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/zzzzzhxxx/article/details/134095703?spm=1001.2014.3001.5502

如果觉得作者写的不错,求给博主一个大大的点赞支持一下,你们的支持是我更新的最大动力!

如果觉得作者写的不错,求给博主一个大大的点赞支持一下,你们的支持是我更新的最大动力!

如果觉得作者写的不错,求给博主一个大大的点赞支持一下,你们的支持是我更新的最大动力!

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

计算机网络原理基础

目录 前言&#xff1a; 1.网络发展史 2.网络通信基础 2.1IP地址 2.1.1定义 2.1.2格式 2.2端口号 2.2.1定义 2.2.2格式 2.3协议 2.3.1定义 2.3.2作用 2.3.3分层 2.4五元组 2.4.1定义 2.4.2组成 3.TCP/IP五层网络模型 3.1模型概念 3.2模型构成 3.3网络分层对应…

aspose-words基础功能演示

我们在Aspose.Words中使用术语“渲染”来描述将文档转换为文件格式或分页或具有页面概念的介质的过程。我们正在讨论将文档呈现为页面。下图显示了 Aspose.Words 中的渲染情况。 Aspose.Words 的渲染功能使您能够执行以下操作: 将文档或选定页面转换为 PDF、XPS、HTML、XAML、…

【数据结构与算法】(5)基础数据结构之队列 链表实现、环形数组实现详细代码示例讲解

目录 2.4 队列1) 概述2) 链表实现3) 环形数组实现 2.4 队列 1) 概述 计算机科学中&#xff0c;queue 是以顺序的方式维护的一组数据集合&#xff0c;在一端添加数据&#xff0c;从另一端移除数据。习惯来说&#xff0c;添加的一端称为尾&#xff0c;移除的一端称为头&#xf…

保姆级系列:各种打印机驱动的安装和使用

保姆级系列&#xff1a;各种打印机驱动的安装和使用 1.介绍2.下载3.安装4.实践教程5.总结 1.介绍 市面上打印机品牌和型号众多&#xff0c;打印机接口目前主要分为如下几种&#xff1a; 逻辑端口&#xff1a;TCP/IP、WSD、USB、LPT、COM 物理端口&#xff1a;RJ45、DB15母、US…

BFS——双向广搜+A—star

有时候从一个点能扩展出来的情况很多&#xff0c;这样几层之后搜索空间就很大了&#xff0c;我们采用从两端同时进行搜索的策略&#xff0c;压缩搜索空间。 190. 字串变换(190. 字串变换 - AcWing题库) 思路&#xff1a;这题因为变化规则很多&#xff0c;所以我们一层一层往外…

19113133262(微信同号)2024年光电信息与机器视觉国际学术会议(ICOIMV 2024)

【征稿进行时|见刊、检索快速稳定】2024年光电信息与机器视觉国际学术会议(ICOIMV 2024) 2024 International Conference Optoelectronic Information and Machine Vision(ICOIMV 2024) 一、【会议简介】 本次会议的主题为“光电信息与机器视觉的未来发展”。围绕这一主题&…

06 - python操作xml

认识XML 与HTML很像&#xff0c;是一种将数据存储在标记之间的标记语言&#xff0c;用户可以自定义自己的标记。 XML文件可以表示称为&#xff1a;XML树。这个XML树从根元素开始&#xff0c;根元素进一步分支到子元素。XML文件的每个元素都是XML树的一个节点&#xff0c;没有…

kafka-splunk数据通路实践

目的&#xff1a; 鉴于目前网络上没有完整的kafka数据投递至splunk教程&#xff0c;通过本文操作步骤&#xff0c;您将实现kafka数据投递至splunk日志系统 实现思路&#xff1a; 创建kafka集群部署splunk&#xff0c;设置HTTP事件收集器部署connector服务创建connector任务&a…

Linux文件编译

目录 一、GCC编译 1.直接编译 2.分步编译 预处理&#xff1a; 编译&#xff1a; 汇编&#xff1a; 链接&#xff1a; 3.多文件编译 4.G 二、Make 1.概述 2.使用步骤 3.makefile创建规则 3.1一个基本规则 3.2两个常用函数 4.示例文件 三、GDB 示例&#xff1a;…

优量汇的广告填充率有多高?

一些量级小的媒体在广告变现初期&#xff0c;以接入广告联盟为主&#xff0c; 广告填充率是衡量广告平台表现和广告变现效率的重要指标。 当用户打开APP时&#xff0c;开发者收到一个对应的广告请求。这个广告请求可以被填充或不被填充&#xff0c;广告填充率广告填充数/广告请…

代码随想录算法训练营第27天| Leetcode 39. 组合总和、40.组合总和II、131.分割回文串

目录 Leetcode 39. 组合总和 Leetcode 40.组合总和II Leetcode 131.分割回文串 Leetcode 39. 组合总和 题目链接&#xff1a;Leetcode 39. 组合总和 题目描述&#xff1a;给定一个无重复元素的数组 candidates 和一个目标数 target &#xff0c;找出 candidates 中所有可以使…

070:vue+cesium: 利用canvas设置径向渐变色材质

相关API参考: https://cesium.com/learn/cesiumjs/ref-doc/ColorMaterialProperty.html 第070个 点击查看专栏目录 本示例的目的是介绍如何在vue+cesium中设置线性渐变色的材质,这里使用canvas的辅助方法。 直接复制下面的 vue+cesium源代码,操作2分钟即可运行实现效果. …