LintCode 12.带最小值操作的栈(两种方法实现)
题目描述
实现一个带有取最小值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.带最小值操作的栈(两种方法实现)的更多相关文章
- lintcode12 带最小值操作的栈
实现一个带有取最小值min方法的栈,min方法将返回当前栈中的最小值. 你实现的栈将支持push,pop 和 min 操作,所有操作要求都在O(1)时间内完成. 建一个栈helpStack,用来存放从 ...
- LintCode-12.带最小值操作的栈
带最小值操作的栈 实现一个带有取最小值min方法的栈,min方法将返回当前栈中的最小值. 你实现的栈将支持push,pop 和 min 操作,所有操作要求都在O(1)时间内完成. 注意事项 如果堆栈中 ...
- Linux 下操作GPIO(两种方法,驱动和mmap)(转载)
目前我所知道的在Linux下操作GPIO有两种方法: 1.编写驱动,这当然要熟悉Linux下驱动的编写方法和技巧,在驱动里可以使用ioremap函数获得GPIO物理基地址指针,然后使用这个指针根据io ...
- Linux 下操作gpio(两种方法,驱动和mmap)
目前我所知道的在linux下操作GPIO有两种方法: 1. 编写驱动,这当然要熟悉linux下驱动的编写方法和技巧,在驱动里可以使用ioremap函数获得GPIO物理基地址指针,然后使用这个指针根据 ...
- 带最小值操作的栈 · Min Stack
[抄题]: 实现一个带有取最小值min方法的栈,min方法将返回当前栈中的最小值. 你实现的栈将支持push,pop 和 min 操作,所有操作要求都在O(1)时间内完成. [思维问题]: [一句话思 ...
- [LintCode] 带最小值操作的栈
class MinStack { public: MinStack() { // do initialization if necessary } void push(int number) { // ...
- Mysql使用binlog恢复数据解决误操作问题的两种方法
为保证没有其他参数配置影响,重新安装配置了一台最小化安装的CentOS7虚拟机 1. 基础知识 安装mysql5.6数据库Mysql binlog初步理解 2. 配置mysql 开启binlog.修 ...
- Android开发12——Andorid中操作数据库的insert的两种方法以及nullColumnHack
一.发现问题 先看两种方法插入数据 public void save(Person p){ SQLiteDatabase db = dbHelper.getWritableDatabase(); db ...
- PHP生成带logo图像二维码的两种方法
本文主要和大家分享PHP生成带logo图像二维码的两种方法,主要以文字和代码的形式和大家分享,希望能帮助到大家. 一.利用Google API生成二维码Google提供了较为完善的二维码生成接口,调用 ...
随机推荐
- ref 和 out 的区别
ref和out都是C#中的关键字,所实现的功能也差不多,都是指定一个参数按照引用传递. 对于编译后的程序而言,它们之间没有任何区别,也就是说它们只有语法区别. 总结起来,他们有如下语法区别: 1.re ...
- onload与ready差异
window.onload: 等所有资源加载完document.ready: DOM树构建完资源还没加载完 应该使用ready保证用户体验.否则当网站有很多图片资源时要很长时间才能加载完这段时间内Js ...
- Android存储扩展学习-----应用的清除数据和清除缓存
前几天和朋友聊到了APP清除数据这块,聊到了清除数据都会清掉哪些数据,我们每个人的手机在”设置–>应用管理”里面,选择任意一个App,都会看到两个按钮,一个是清除缓存,另一个是清除数据,那么当我 ...
- CodeMirror教程,CodeMirrorAPI中文信息
<html> <head> <link rel="stylesheet" href="codemirror.css"> ...
- solidity语言8
输入参数 pragma solidity ^0.4.16; contract Simple { function taker(uint _a, uint _b) public pure { // do ...
- Flume的load-balance、failover
配置flume集群参考https://www.cnblogs.com/jifengblog/p/9277793.html load-balance负载均衡 介绍 负载均衡是用于解决一台机器(一个进程) ...
- C# 64位win7下DllImport LoadLibrary函数失败 z
[DllImport["kernel32.dll"]] static extern IntPtr LoadLibrary(string lpFileName); public vo ...
- mysql修改管理员密码
mysql修改管理员密码杀掉mysql进程kill `cat /data/mysqldata/3306/mysql.pid`禁止连接禁止验证方式启动mysqlmysqld_safe --default ...
- 我的HTML总结之HTML发展史
HTML是Web统一语言,这些容纳在尖括号里的简单标签,构成了如今的Web. 1991年,Tim Berners-Lee编写了一份叫做“HTML标签”的文档,里面包含了大约20个用来标记网页的HTML ...
- POJ-2976 Dropping tests---二分最大化平均值
题目链接: https://cn.vjudge.net/problem/POJ-2976 题目大意: 给定n个二元组(a,b),扔掉k个二元组,使得剩下的a元素之和与b元素之和的比率最大 解题思路: ...