每日一练:LeeCode-503. 下一个更大元素 II (中)【单调栈】

本文是力扣LeeCode-503. 下一个更大元素 II 学习与理解过程,本文仅做学习之用,对本题感兴趣的小伙伴可以出门左拐LeeCode。

给定一个循环数组 nums ( numsnums.length - 1 的下一个元素是 nums0 ),返回 nums 中每个元素的 下一个更大元素 。

数字 x 的 下一个更大的元素 是按数组遍历顺序,这个数字之后的第一个比它更大的数,这意味着你应该循环地搜索它的下一个更大的数。如果不存在,则输出 -1 。

示例 1:

输入: nums = 1,2,1

输出: 2,-1,2

解释: 第一个 1 的下一个更大的数是 2;

数字 2 找不到下一个更大的数;

第二个 1 的下一个最大的数需要循环搜索,结果也是 2。

示例 2:

输入: nums = 1,2,3,4,3

输出: 2,3,4,-1,4


单调栈法 :单调栈法是我们刷算法必备的黄金公式

对单调栈的理解我参考 负雪明烛博主的讲解

在实现上,我们可以使用**「单调栈」**来实现,单调栈是说 栈里面的元素从栈底到栈顶单调递增或者单调递减 的(类似于汉诺塔)。本题应该用个「单调递减栈」来实现。
建立「单调递减栈」,并对原数组遍历一次:
如果栈为空,则把当前元素放入栈内;
如果栈不为空,则需要判断当前元素和栈顶元素的大小:
如果当前元素比栈顶元素大:说明当前元素是前面一些元素的「下一个更大元素」,则逐个弹出栈顶元素,直到当前元素比栈顶元素小为止。
如果当前元素比栈顶元素小:说明当前元素的「下一个更大元素」与栈顶元素相同,则把当前元素入栈。

模版 :以 下一个更大元素 I 为例

java 复制代码
class Solution {
    public int[] nextGreaterElement(int[] nums1, int[] nums2) {
        Map<Integer, Integer> map = new HashMap<Integer, Integer>();
        Deque<Integer> stack = new ArrayDeque<Integer>();
        for (int i = nums2.length - 1; i >= 0; --i) {
            int num = nums2[i];
            while (!stack.isEmpty() && num >= stack.peek()) {
                stack.pop();
            }
            map.put(num, stack.isEmpty() ? -1 : stack.peek());
            stack.push(num);
        }
        int[] res = new int[nums1.length];
        for (int i = 0; i < nums1.length; ++i) {
            res[i] = map.get(nums1[i]);
        }
        return res;
    }
}

思路

考虑到本题的实际情况,需要回过头来遍历数组,这里牵扯到 循环数组 的概念。
循环数组数组的最后一个元素下一个元素是数组的第一个元素 ,形状类似于「环」 。我们只需要使用 取模运算 % 可以把下标i映射到数组 nums长度的 0−N内。

未通过版本

参照下一个更大元素 I 的解决方法并结合循环数据,我的代码如下:

java 复制代码
    public int[] nextGreaterElements(int[] nums) {
        int[] result = new int[nums.length];
        Map<Integer,Integer> map = nextGreaterElementsHelper(nums);
        for(int i=0;i<nums.length;i++){
            result[i] = map.get(nums[i]);
        }
        return result;
    }

    private Map<Integer,Integer> nextGreaterElementsHelper(int[] nums){
        int len = nums.length;
        Map<Integer,Integer> map = new HashMap<>();
        Stack<Integer> stack = new Stack<>();
        for(int j=2*len-1;j>=0;j--){	//逻辑环,取模运算
            while(!stack.isEmpty()&&stack.peek()<=nums[j%len]){
                stack.pop();
            }
            map.put(nums[j%len],stack.isEmpty()?-1:stack.peek());
            stack.push(nums[j%len]);
        }
        return map;
    }

提交报错


从1开始就出问题了,debug了半天,没看出问题。有大佬帮我看出问题吗?

代码实现

java 复制代码
class Solution {
    public int[] nextGreaterElements(int[] nums) {
        int n = nums.length;
        int[] ret = new int[n];
        Arrays.fill(ret, -1);
        Deque<Integer> stack = new LinkedList<Integer>();
        for (int i = 0; i < n * 2 - 1; i++) {
            while (!stack.isEmpty() && nums[stack.peek()] < nums[i % n]) {
                ret[stack.pop()] = nums[i % n];
            }
            stack.push(i % n);
        }
        return ret;
    }
}

大家有更好的方法,请不吝赐教。

相关推荐
故渊at3 分钟前
第二板块:Android 四大组件标准化学理 | 第七篇:Activity 页面载体与任务栈算法
android·算法·生命周期·activity·任务栈
兰令水10 分钟前
leecodecode【区间DP+树形DP】【2026.6.10打卡-java版本】
java·算法·leetcode
weixin1997010801635 分钟前
[特殊字符] 1688开放平台API Sign签名算法详解(Java / Python / PHP 实现)
java·python·算法
未若君雅裁1 小时前
JVM 垃圾回收算法与分代回收机制
java·jvm·算法
智者知已应修善业1 小时前
【51单片机初始化D5-D8亮,每按键按下D1到D4全亮,再按下恢复,如此循环】2024-3-26
c++·经验分享·笔记·算法·51单片机
8Qi81 小时前
LeetCode 4:寻找两个正序数组的中位数 —— 二分查找法
java·算法·leetcode·职场和发展·二分查找
8Qi81 小时前
LeetCode 32:最长有效括号 —— 栈 + 标记法 题解
java·数据结构·算法·leetcode·职场和发展··括号匹配
机器学习之心1 小时前
198种组合算法+优化CNN-LSTM+SHAP分析+新数据预测+多输出!深度学习可解释分析,强烈安利,粉丝必备
深度学习·算法·cnn-lstm·shap分析·198种组合算法
Tairitsu_H1 小时前
[LC优选算法#3] 滑动窗口 | 将x减到0的最⼩操作数 | ⽔果成篮 | 字⺟异位词
c++·算法·leetcode·滑动窗口