首页 > 编程语言 >面试算法:计算堆栈当前元素的最大值

面试算法:计算堆栈当前元素的最大值

时间:2023-06-14 11:05:30浏览次数:53  
标签:maxStack 压入 最大值 元素 面试 ms 堆栈 stack


更详细的讲解和代码调试演示过程,请参看视频
如何进入google,算法面试技能全面提升指南

有一道堆栈相关算法题,我被面试过两次以上,看似其在算法面试中出现的概率很高,由此值得我们好好分析下。题目是这样的:

对于堆栈的常用操作有, pop 弹出堆栈顶部的元素;push 向堆栈压入一个元素;peek 获得堆栈顶部的元素值,但不弹出堆栈。现在要去你增加一个操作max, 它的作用是返回堆栈当前所有元素中值最大的那个,例如堆栈当前元素有:
stack: 5,4,2,3
那么max() 返回的值就是5. 假设向堆栈继续推入元素后,情况如下:

stack: 5, 4, 2, 3, 6, 1, 10, 8

那么调用max() 得到的元素为10.

假设我们现在我们采取pop操作,弹出顶部两个元素,那么堆栈情况如下:

stack: 5, 4, 2, 3, 6, 1

显然此时max 操作返回的元素是6。

请给出一个时间复杂度为O(1)的算法,实现max操作。

这道题一个麻烦处在于,如果堆栈仅仅是压入元素,那么返回最大值是很容易的,只要把当前压入元素的最大值记下来就可以了。问题在于,堆栈还有弹出操作,当弹出后,堆栈当前的最大元素可能就会不断的发生变化。如果是压入操作和弹出操作交替进行的话,那么情况就更复杂了。

解决这个问题的算法如下:
1, 每次压入元素是,用一个变量maxVal, 记录堆栈当前元素的最大值。
2, 创建一个新堆栈maxStack,当压入元素的值大于当前元素的最大值时,把该元素压入maxStack.
3, 弹出一个元素时,如果弹出的元素是当前最大值,那么把maxStack顶部的元素也弹出,然后把maxValue设置成maxStack的顶部元素。
4,执行max()操作时,可以直接返回maxValue, 或是maxStack堆栈的顶部元素。

不难看出,上述操作使得元素压入时,maxValue 与 maxStack顶部元素保持一致,当元素弹出时,如果弹出的是当前最大值,那么步骤3把maxValue的值设置成maxStack的顶部元素,因此,无论是压入还是弹出操作,maxValue与maxStack的始终保持一致。

同时,当只有压入元素大于当前堆栈元素的最大值时,新元素才会压入maxStack,这就保证了maxStack堆栈顶部的元素就是当前堆栈中所有元素最大的一个。

上面算法,执行max()操作时,只需要把maxStack顶部元素弹出或直接返回maxValue,因此时间复杂度是O(1), 在特殊情况下,例如所有元素是升序压入堆栈的,比如压入的元素为:1,2, 3, 4.那么这些元素除了压入正常堆栈外,还得全部压入maxStack, 因此算法的空间复杂度是O(N).

我们看看具体代码的实现:

import java.util.Stack;


public class MaxStack {
    private Stack<Integer> stack = new Stack<Integer>();
    private int maxVal = Integer.MIN_VALUE;
    private Stack<Integer> maxStack = new Stack<Integer>();

    public void push(int val) {
        if (val >= maxVal) {
            maxVal = val;
            maxStack.push(maxVal);
        }

        stack.push(val);
    }

    public int peek() {
        return stack.peek();
    }

    public int pop() {
        if (stack.peek() == maxVal) {
            maxStack.pop();
        }

        maxVal = maxStack.peek();

        return stack.pop();
    }

    public int max() {
        return maxStack.peek();
    }
}

代码实现简单,基本上根据算法步骤来实现,我们再看看主入口处的代码:

public class StackAndQuque {
    public static void main(String[] args) {
        MaxStack ms = new MaxStack();
        ms.push(5);
        ms.push(4);
        ms.push(2);
        ms.push(3);

        System.out.println("Max Val in stack is : " + ms.max());

        ms.push(6);
        ms.push(1);
        ms.push(10);
        ms.push(8);
        System.out.println("Max Val in stack is : " + ms.max());

        ms.pop();
        ms.pop();
        System.out.println("Max Val in stack is : " + ms.max());

        ms.push(7);
        System.out.println("Max Val in stack is : " + ms.max());

    }
}

运行结果如下:

Max Val in stack is : 5
Max Val in stack is : 10
Max Val in stack is : 6
Max Val in stack is : 7

稍微检测下便可以发现,代码给出的结果是正确的,更多更详实的讲解和调试演示过程,请大家参考视频。

更多技术信息,包括操作系统,编译器,面试算法,机器学习,人工智能,请关照我的公众号:

面试算法:计算堆栈当前元素的最大值_Stack


标签:maxStack,压入,最大值,元素,面试,ms,堆栈,stack
From: https://blog.51cto.com/u_16160261/6476219

相关文章

  • 算法面试:单向链表节点的奇偶排序。
    给定一个单项链表,要求实现一个算法,把链表分成两部分,前一部分全是下标为偶数的节点,后一部分全是下标为奇数的节点,例如给定链表为下图的第一个队列,要求编写一个算法,将链表转换为第二个队列:要求算法不能分配多余的内存,同时,在操作链表是,不能更改节点内部的内容,只能更改节点的next指针......
  • 面试算法:获取重合列表的第一个相交节点
    给定两个单向链表,这两个链表有可能会有重叠,情况如下:两个单向链表从节点5开始重合,要求给定一个空间复杂度为O(1)的算法,返回两个链表相交时的第一个节点。依据上图,也就是返回节点5.首先我们需要做的是,确保给定的两个单向链表,他们是相交的。这个很好确定,只要从头遍历两个链表,如果他们......
  • 算法面试之道:在O(1)的时间内删除单链接链表的指定节点
    对于一个单项链接的链表,给定其中某个任意节点,要求在O(1)的时间复杂度内删除该节点。表面上看起来,似乎不可能做到,因为如果要求时间复杂度是O(1)的话,那意味着,算法实现中,不得包含有任何循环或是对链表的整体遍历。但问题在于,要删除某个指定节点,我们需要通过遍历,找到该节点的前节点,然......
  • 面试算法:波浪型数组的快速排序法
    波浪型数组是具备这样特性的数组,它的元素先是递增,到达某个点后开始低贱,接着达到某个点时又递增,接着右递减,以下数组就是一个波浪型数组:A:57,131,493,294,221,339,418,452,442,190A[0]A[1]A[2]A[3]A[4]A[5]A[6],A[7],A[8],A[9]不难发现,A[0]-A[4]是一个波浪,因为......
  • 2373.矩阵中的局部最大值
    问题描述2373.矩阵中的局部最大值(Easy)给你一个大小为nxn的整数矩阵grid。生成一个大小为(n-2)x(n-2)的整数矩阵maxLocal,并满足:maxLocal[i][j]等于grid中以i+1行和j+1列为中心的3x3矩阵中的最大值。换句话说,我们希望找出grid中每个......
  • C++面试八股文:C++中,函数的参数应该传值还是传引用?
    C++面试八股文:C++中,函数的参数应该传值还是传引用?某日二师兄参加XXX科技公司的C++工程师开发岗位第8面:面试官:C++中,函数的参数应该传值还是传引用?二师兄:要看参数的用途。如果是出参,必须传引用。如果是入参,主要考虑参数类型的大小,来决定传值还是传引用。面试官:为什么不使用......
  • C++面试八股文:什么是RAII?
    某日二师兄参加XXX科技公司的C++工程师开发岗位第13面:面试官:什么是RAII?二师兄:RAII是ResourceAcquisitionIsInitialization的缩写。翻译成中文是资源获取即初始化。面试官:RAII有什么特点和优势?二师兄:主要的特点是,在对象初始化时获取资源,在对象析构时释放资源。这种技术可以......
  • C++面试八股文:什么是RAII?
    某日二师兄参加XXX科技公司的C++工程师开发岗位第13面:面试官:什么是RAII?二师兄:RAII是ResourceAcquisitionIsInitialization的缩写。翻译成中文是资源获取即初始化。面试官:RAII有什么特点和优势?二师兄:主要的特点是,在对象初始化时获取资源,在对象析构时释放资源。这种技术可以......
  • #yyds干货盘点# LeetCode程序员面试金典:被围绕的区域
    题目:给你一个mxn的矩阵board,由若干字符'X'和'O',找到所有被'X'围绕的区域,并将这些区域里所有的 'O'用'X'填充。 示例1:输入:board=[["X","X","X","X"],["X","O","O",&quo......
  • Java面试笔记202306
    Java基础ArrayListArrayList底层数据是动态数组,初始长度为10,每次扩容为原来的1.5倍。扩容流程:首先会创建一个新的长度的数组,然后使用Arrays.copyOf()方法将旧的数组中的元素复制到新的数组中,最后会将新插入的数据插入到新的数组中。IO和NIO的区别io指的是io流。可以实现数......