题目描述

实现一个带有取最小值min方法的栈,min方法将返回当前栈中的最小值。

你实现的栈将支持push,pop 和 min 操作,所有操作要求都在O(1)时间内完成。

样例

如下操作:push(1),pop(),push(2),push(3),min(), push(1),min() 返回 1,2,1

先审题,返回值为1,2,1,分别是 pop(),min(),min() 这三个操作返回的值;在一般栈结构的基础上实现min方法来返回当前栈中的最小值,

我们可以增加一个辅助栈结构,用来记录当前栈的最小值。

手动推导一遍样例(方法1)

操作

当前栈:datastack

辅助栈:minstack

返回值

push(1)

1

1

pop()

1

push(2)

2

2

push(3)

2,3

2,2

min()

2,3

2,2

2

push(1)

2,3,1

2,2,1

min()

2,3,1

2,2,1

1

从上表的推导过程来看,当前栈的 push(),pop()操作与一般栈无区别,新添加的 min() 操作每次返回的是当前栈中的最小值元素,即辅助栈的栈顶元素;

而增加的辅助栈在 push() 操作上做了一点改变,每次压栈时,若当前元素 <= 辅助栈栈顶元素时,则将当前元素压入栈,否则,继续压入辅助栈栈顶元素;

实现代码:

 import java.util.Stack;

 public class MinStack {
public static void main(String[] args) {
MinStack stack = new MinStack();
stack.push(1);
stack.pop();
stack.push(2);
stack.push(3);
stack.min();
stack.push(1);
stack.min(); } Stack<Integer> datastack = new Stack<Integer>();
Stack<Integer> minstack = new Stack<Integer>(); /*当前栈压栈,1)最小值栈为空时,将该元素记录为最小值栈栈顶元素;2)最小值栈非空时,将该元素与最小值栈栈顶元素比较,谁小谁入栈;*/
public void push(int num) {
int top;
datastack.push(num);
if(!minstack.empty())
top = minstack.peek();
else
top = num; if(num > top)
minstack.push(top);
else
minstack.push(num); //System.out.println("push"+num+"时当前栈的栈顶元素为:"+datastack.peek());
//System.out.println("push"+num+"时最小值栈的栈顶元素为:"+minstack.peek()); }
/*返回当前栈栈顶元素,当前栈与最小值栈同时出栈;*/
public int pop() {
if(!datastack.isEmpty()) {
int data = datastack.peek();
datastack.pop();
minstack.pop();
//return data;
System.out.println("当前栈中的最小值为:"+ data);
}
return 0;
}
/*判断当前栈是否非空,返回当前栈中的最小值,即最小值栈的栈顶元素;*/
public int min() {
if(!datastack.isEmpty()) {
//return minstack.peek();
System.out.println("当前栈中的最小值为:"+minstack.peek());
}
return 0;
}
}

手动推导一遍样例(方法2)

操作

当前栈:datastack

辅助栈:minstack

返回值

push(1)

1

1

pop()

1

push(2)

2

2

push(3)

2,3

2

min()

2,3

2

2

push(1)

2,3,1

2,1

min()

2,3,1

2,1

1

方法2与方法1区别在于,第一点:方法2使用LinkedList数据结构来实现栈,而不是如方法1中直接使用Stack来实现栈;

第二点:方法2的辅助栈的 push() 操作只在栈为空或者当前元素 < 辅助栈栈顶元素时,才压栈;pop() 操作时,只有在栈顶元素与当前栈出栈元素相同时才出栈。

实现代码:

import java.util.LinkedList;

public class MinStack {
public static void main(String[] args) {
MinStack stack = new MinStack();
stack.push(1);
stack.pop();
stack.push(2);
stack.push(3);
stack.min();
stack.push(1);
stack.min(); }
LinkedList datastack = new LinkedList<Integer>();
LinkedList minstack = new LinkedList<Integer>(); /* 当前栈压栈,最小值栈为空时或者该元素比最小值栈栈顶元素小时压栈 */
public void push(int num) {
datastack.addFirst(num);
if(minstack.isEmpty()||num <= (int)minstack.getFirst()) {
minstack.addFirst(num);
}
} /* 判断当前栈非空,则出栈,若出栈元素与最小值栈栈顶元素相同,则也须出栈 */
public int pop() {
int data;
if(!datastack.isEmpty()) {
data = (int)datastack.removeFirst();
if(data == (int)minstack.getFirst()) {
minstack.removeFirst();
}
//return data;
System.out.println(data);
}
return 0;
} /* 返回最小值栈的栈定元素即所求的最小元素 */
public int min() {
if(!datastack.isEmpty()) {
return (int)minstack.getFirst();
//System.out.println((int)minstack.getFirst());
}
return 0;
}
}

总结:方法1与方法2在LintCode上均提交成功,运行时间分别为 2042ms 和 2088ms .

直接使用Stack类实现栈,理解简单易上手,重点掌握其 push()、pop()、peek()等方法;而使用LinkedLIst类来实现栈,这里只是为了加强LinkedList的练习,

重点掌握其 addFirst()、getFirst()、removeFirst() 等方法;同时还可以用ArrayList类来实现栈。

LintCode 12.带最小值操作的栈(两种方法实现)的更多相关文章

  1. lintcode12 带最小值操作的栈

    实现一个带有取最小值min方法的栈,min方法将返回当前栈中的最小值. 你实现的栈将支持push,pop 和 min 操作,所有操作要求都在O(1)时间内完成. 建一个栈helpStack,用来存放从 ...

  2. LintCode-12.带最小值操作的栈

    带最小值操作的栈 实现一个带有取最小值min方法的栈,min方法将返回当前栈中的最小值. 你实现的栈将支持push,pop 和 min 操作,所有操作要求都在O(1)时间内完成. 注意事项 如果堆栈中 ...

  3. Linux 下操作GPIO(两种方法,驱动和mmap)(转载)

    目前我所知道的在Linux下操作GPIO有两种方法: 1.编写驱动,这当然要熟悉Linux下驱动的编写方法和技巧,在驱动里可以使用ioremap函数获得GPIO物理基地址指针,然后使用这个指针根据io ...

  4. Linux 下操作gpio(两种方法,驱动和mmap)

    目前我所知道的在linux下操作GPIO有两种方法: 1.  编写驱动,这当然要熟悉linux下驱动的编写方法和技巧,在驱动里可以使用ioremap函数获得GPIO物理基地址指针,然后使用这个指针根据 ...

  5. 带最小值操作的栈 · Min Stack

    [抄题]: 实现一个带有取最小值min方法的栈,min方法将返回当前栈中的最小值. 你实现的栈将支持push,pop 和 min 操作,所有操作要求都在O(1)时间内完成. [思维问题]: [一句话思 ...

  6. [LintCode] 带最小值操作的栈

    class MinStack { public: MinStack() { // do initialization if necessary } void push(int number) { // ...

  7. Mysql使用binlog恢复数据解决误操作问题的两种方法

    为保证没有其他参数配置影响,重新安装配置了一台最小化安装的CentOS7虚拟机 1. 基础知识
 安装mysql5.6数据库Mysql binlog初步理解 2. 配置mysql 开启binlog.修 ...

  8. Android开发12——Andorid中操作数据库的insert的两种方法以及nullColumnHack

    一.发现问题 先看两种方法插入数据 public void save(Person p){ SQLiteDatabase db = dbHelper.getWritableDatabase(); db ...

  9. PHP生成带logo图像二维码的两种方法

    本文主要和大家分享PHP生成带logo图像二维码的两种方法,主要以文字和代码的形式和大家分享,希望能帮助到大家. 一.利用Google API生成二维码Google提供了较为完善的二维码生成接口,调用 ...

随机推荐

  1. Git和GitHub在线学习资源整理(转)

    原文地址:http://blog.csdn.net/duqi_2009/article/details/12646711 电子书 GotGitHub Git Workflow 文章 GitHub Fu ...

  2. scss-注释

    在scss中有两种注释方式 原生css的注释多行注释: /* *  注释的内容 */ 单行注释:// 注释内容一致延续到行末. 在尽可能的情况下,多行注释会被保留在输出的CSS中,而单行注释会被删除.

  3. 给HTML拍个照(如何将html元素转成图片)

    本文主要介绍一款好用的库,如何将HTML生成图片. 1.简述 最近在做的项目中,需要将界面转换成模板保存下来,本来想使用自适应布局完成,但是页面较复杂,模板较多,生成的模板使用过多的HTML标签,于是 ...

  4. Web站点如何防范XSS、CSRF、SQL注入攻击

    XSS跨站脚本攻击 XSS跨站脚本攻击指攻击者在网页中嵌入客户端脚本(例如JavaScript),当用户浏览此网页时,脚本就会在用户的浏览器上执行,从而达到攻击者的目的,比如获取用户的Cookie,导 ...

  5. Lucene 初识

    因为业务需要,虽然自己不是专门写搜索的,但是需要自己拼一些搜索条件去调用搜索的接口,而之前看的JVM crash里也涉及到了Lucene,所以大概了解一下. 参考文档: http://www.itey ...

  6. 【数据库】2.0 MySQL入门学习(二)——如何获得MySQL以及MySQL安装

    1.0 如何获得MySQL: www.oracle.com https://dev.mysql.com/downloads/ 2.0 例如进入Oracle官网,找到MySQL: 进入页面后,切换到“资 ...

  7. 【代码笔记】Java基础:Java的方法和类

    面向过程与面向对象都是我们编程中,编写程序的一种思维方式.例如:公司打扫卫生(擦玻璃.扫地.拖地.倒垃圾等), 按照面向过程的程序设计方式会思考“打扫卫生我该怎么做,然后一件件的完成”,最后把公司卫生 ...

  8. easyui datagrid 获取行数据某个字段

    首先要能获取datagrid 的row对象 即:var row = $('#datagrid').datagrid('getData').rows[index]; 之后我们就可以通过类似row.nam ...

  9. Android 隐式 Intent 跳转注意事项

    前几天正好看到<阿里巴巴 Android 开发手册>中提到的: “Activity 间通过隐式 Intent 的跳转,在发出 Intent 之前必须通过 resolveActivity 检 ...

  10. 防反编译、混淆文件proguard.cfg与proguard-project.txt详解

    在新版本的ADT创建项目时,混码的文件不再是proguard.cfg,而是project.properties和proguard-project.txt. 如果需要对项目进行全局混码,只需要进行一步操 ...