Leetcode 496 下一个更大元素 I

个人感觉自己总结的思路没有官方解答好,所以思路都是复制的官方解答。其实思路不难理解,重要的是思想。

题目描述

nums1 中数字 x下一个更大元素 是指 xnums2 中对应位置 右侧第一个x 大的元素。

给你两个 没有重复元素 的数组 nums1nums2 ,下标从 0 开始计数,其中nums1nums2 的子集。

对于每个 0 <= i < nums1.length ,找出满足 nums1[i] == nums2[j] 的下标 j ,并且在 nums2 确定 nums2[j] 的 下一个更大元素 。如果不存在下一个更大元素,那么本次查询的答案是 -1

返回一个长度为 nums1.length 的数组 ans 作为答案,满足 ans[i] 是如上所述的 下一个更大元素

案例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

方法一:暴力

我们可以暴力地逐个计算 nums1中的每个元素值 nums1[i]nums2中对应位置的右边的第一个比 nums1[i] 大的元素值。具体地,我们使用如下方法:

  • 初始化与 nums1 等长的查询数组 res

  • 遍历 nums1中的所有元素,不妨设当前遍历到元素为 nums1[i]:

    • 从前向后遍历 nums2中的元素,直至找到 nums2[j]=nums1[i]
    • j+1 开始继续向后遍历,直至找到 nums2[k]>nums2[j]
    • 如果找到了 nums2[k],则将 res[i] 置为 nums2[k],否则将 res[i] 置为 1

    数组 res即为最终结果。

    class Solution {
        public int[] nextGreaterElement(int[] nums1, int[] nums2) {
            int length = nums2.length;
            int[] ans = new int[nums1.length];
            for (int i = 0; i < nums1.length; i++) {
                int currNums1 = nums1[i];
                int j = 0;
                while (nums2[j]!=currNums1) {
                    j++;
                }
                int index = j + 1;
                while (index < length && nums2[index] <= currNums1) {
                    ++index;
                }
                if (index < length) {
                    ans[i] = nums2[index];
                } else {
                    ans[i] = -1;
                }
            }
            return ans;
        }
    }
    

方法二:单调栈

一旦要求下一个更大的元素,就是用单调栈解

我们可以先预处理 nums2,使查询 nums1中的每个元素在 nums2n 中对应位置的右边的第一个更大的元素值时不需要再遍历 nums2。于是,我们将题目分解为两个子问题:

  • 1 个子问题:如何更高效地计算 nums2 中每个元素右边的第一个更大的值;
  • 2 个子问题:如何存储第 1 个子问题的结果。

我们可以使用单调栈来解决第 1 个子问题。倒序遍历 nums2,并用单调栈中维护当前位置右边的更大的元素列表,从栈底到栈顶的元素是单调递减的。

具体地,每次我们移动到数组中一个新的位置 i,就将当前单调栈中所有小于 nums2[i] 的元素弹出单调栈,当前位置右边的第一个更大的元素即为栈顶元素,如果栈为空则说明当前位置右边没有更大的元素。随后我们将位置 i 的元素入栈。

因为题目规定了 nums2 是没有重复元素的,所以我们可以使用哈希表来解决第 2 个子问题,将元素值与其右边第一个更大的元素值的对应关系存入哈希表。

class Solution {
    public int[] nextGreaterElement(int[] nums1, int[] nums2) {
        int length = nums2.length;
        int[] ans = new int[nums1.length];
        for (int i = 0; i < nums1.length; i++) {
            int currNums1 = nums1[i];
            int j = 0;
            while (nums2[j]!=currNums1) {
                j++;
            }
            int index = j + 1;
            while (index < length && nums2[index] <= currNums1) {
                ++index;
            }
            if (index < length) {
                ans[i] = nums2[index];
            } else {
                ans[i] = -1;
            }
        }
        return ans;
    }
}
posted @   云小杰  阅读(20)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 记一次.NET内存居高不下排查解决与启示
· DeepSeek 开源周回顾「GitHub 热点速览」
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
点击右上角即可分享
微信分享提示