用O(1)的时间复杂度,找到栈和队列中的最小(大)值
最近刷剑指offer,看到两道编程题,考察在O(1)的复杂度内,找出最值。
觉得很有意思,很有借鉴意义,故记录在此。
需要注意的是,这里所说的O(1) 有个前提, 就是已经通过某种容器的存储方式进行初始化,
不然不可能在还未遍历数据的情况下就定位出最值。
问题1: 重新定义栈的数据结构,实现一个能够在O(1)时间复杂度内求出栈内最小元素的min函数。
思路: 需要一个变量minimum保存目前栈内所有值的最小值,因为最小值是随着出栈,入栈操作变化的,所以一个变量是不够的。
考虑引入辅助栈,辅助栈中保存于数据栈中同步的当前最小值。 即辅助栈栈顶元素为当前数据栈内的最小值。
例如:stack_data中元素为[3,4,1,2] , 则stack_support中为[3,3,1,1]。 当数据栈2出栈,同时辅助栈1出栈, 则剩余中最小值还是辅助栈顶元素1; 数据栈再出栈1,辅助栈也出栈1, 则剩下的数据栈最小元素为辅助栈栈顶元素3.
需要重写栈的push, pop操作。
C++代码:
template <typename T> class NewStack
{
private :
std::stack<T> stack_data;
std::stack<T> stack_support; public: NewStack();
~NewStack(); void push( T value)
{
stack_data.push_back(value); if (stack_support.size()== || stack_support.top()>value)
stack_support.push_back(value);
else:
stack_support.push_back(stack_support.top()); } void pop()
{
if (stack_data.size()> && stack_support.size()>)
{
stack_data.pop_back();
stack_support.pop_back();
} } T min()
{
if (stack_data.size()> && stack_support.size()>)
{
return stack_support.top();
} }
问题2:实现在O(1)时间复杂度内,找出队列中的最小值。
思路:前文中我们实现了栈中O(1)找最小值,因此我们只需要通过两个栈(FILO)实现一个队列(FIFO),就可以实现队列O(1)找到最小值。
即stack1的栈顶作为queue的入口,stack2的栈顶作为queue的出口。
C++代码:两个栈实现一个队列如下所示:
template <typename T> class NewQueue
{
private :
std::stack<T> stack1;
std::stack<T> stack2; public: NewQueue(void);
~NewQueue(void); void append(T value)
{
stack1.push_back(value); } T pop()
{
//如果stack2为空,则从stack1拿元素中入栈到stack中
if (stack2.size()<=)
{
while(stack1.size()>)
{
T element = stack1.top();
stack1.pop_back();
stack2.push_back(element); } }
// 如果已经没有元素可以出栈了
if (stack1.size()==)
throw new exception("queue is empty.") T res = stack2.top();
stack2.pop_back();
return res;
} }
如果要解决问题2, 只需结合代码1和2,在代码2中引入stack_support存放最小值即可:
代码如下:
template <typename T> class NewQueue
{
private :
std::stack<T> stack1;
std::stack<T> stack2;
std::stack<T> stack_support; public: NewQueue(void);
~NewQueue(void); void append(T value)
{
stack1.push_back(value); if (stack_support.size()== || stack_support.top()>value)
stack_support.push_back(value);
else:
stack_support.push_back(stack_support.top()); } T pop()
{
//如果stack2为空,则从stack1拿元素中入栈到stack中
if (stack2.size()<=)
{
while(stack1.size()>)
{
T element = stack1.top();
stack1.pop_back();
stack2.push_back(element); } }
// 如果已经没有元素可以出栈了
if (stack2.size()== && stack_support.size()==)
throw new exception("queue is empty.") T res = stack2.top();
stack2.pop_back(); stack_support.pop_back();
return res; T min()
{
if (stack2.size()> && stack_support.size()>)
{
return stack_support.top();
}
} }
用O(1)的时间复杂度,找到栈和队列中的最小(大)值的更多相关文章
- 剑指offer:按之字形打印二叉树(栈|双向队列+中序遍历)
1. 题目描述 /** 请实现一个函数按照之字形打印二叉树, 即第一行按照从左到右的顺序打印, 第二层按照从右至左的顺序打印, 第三行按照从左到右的顺序打印, 其他行以此类推. */ 2. 双向队列 ...
- 包含min函数的栈、队列
题目:定义栈的数据结构,请在该类型中实现一个能够得到栈/队列的最小元素的min函数.在该栈/队列中,调用min.入栈(入队列)及出栈(出队列)函数的时间复杂度都是O(1). 1. 包含min函数的栈 ...
- Java的栈和队列
package com.ipmotor.sm.db;import java.util.LinkedList;import java.util.Queue;import java.util.Stack; ...
- O(1)时间复杂度求栈中最小元素
import java.util.Stack; /** * 功能:O(1)时间复杂度求栈中最小元素 * 思路:空间换取时间,使用两个栈,stack1栈存储数据,stack2栈存储最小值: * stac ...
- 如何在O(1)时间复杂度获取栈中最大值和最小值
问题描述: 如何在O(1)时间复杂度获取栈中的最大值和最小值? 问题分析: 普通栈规定的push(入栈).pop(出栈).peek(查看栈顶)等操作都只能在栈顶上操作,如果栈中元素是有序的,那么我们就 ...
- 单调栈&单调队列入门
单调队列是什么呢?可以直接从问题开始来展开. Poj 2823 给定一个数列,从左至右输出每个长度为m的数列段内的最小数和最大数. 数列长度:\(N <=10^6 ,m<=N\) 解法① ...
- java——栈和队列 面试题
(1)栈的创建 (2)队列的创建 (3)两个栈实现一个队列 (4)两个队列实现一个栈 (5)设计含最小函数min()的栈,要求min.push.pop.的时间复杂度都是O(1) (6)判断栈的push ...
- 笔试算法题(05):转换BST为双向链表 & 查找栈中的最小元素
出题:把二元查找树转变成排序的双向链表.输入一棵二元查找树,要求将该二元查找树按照中序转换成一个排序的双向链表,要求不能创建任何新的节点,只能调整指针的指向: 分析: 递归的思路,当前节点需要进行的处 ...
- JavaScript 数据结构与算法之美 - 线性表(数组、栈、队列、链表)
前言 基础知识就像是一座大楼的地基,它决定了我们的技术高度. 我们应该多掌握一些可移值的技术或者再过十几年应该都不会过时的技术,数据结构与算法就是其中之一. 栈.队列.链表.堆 是数据结构与算法中的基 ...
随机推荐
- 用ie调试的时候显示:脚本调试程序无法连接到目标进程,已附加调试程序。
解决方案如图所示: 解决方案: 在internet的选项工具中选中高级然后去掉禁止脚本调试的情况:
- cmd中目录的变更
1.在同一个盘符下的目录的切换使用cd加空格加子目录进行切换 2.在不同的盘符下进行切换直接使用在当前目录后面加上其他盘符的名称即可
- POJ 3207 Ikki's Story IV - Panda's Trick
Ikki's Story IV - Panda's Trick Time Limit: 1000MS Memory Limit: 131072K Total Submissions: 7296 ...
- Python计算程序运行时间
方法1 import datetime starttime = datetime.datetime.now() #long running endtime = datetime.datetime.no ...
- EndNote文献管理
一直想写个博客,但是一直没有好好坐下来对自己工作进行一个梳理.从今天开始吧,争取多写一点. 今天,先介绍一下科技论文写作中经常使用的一款软件EndNote,这个软件,掌握它的使用方法后会觉得很方便:但 ...
- dll 学习(一)
DLL(Dynamic Link Library)的概念,你可以简单的把DLL看成一种仓库,它提供给你一些可以直接拿来用的变量.函数或类.在仓库的发展史上经历了"无库-静态链接库-动态链接库 ...
- PNG图片数据解析
PNG是一种非常流行的图片格式,它不仅支持透明效果,而且图片数据经过了压缩处理,所以广泛用于web等应用. PNG的文件格式: PNG文件中的数据,总是以一个固定的8个字节开头: (图片来自http: ...
- 常用的几种 SQLServer 分页查询方式实现
SQLServer 的数据分页: 假设现在有这样的一张表:CREATE TABLE test( id int primary key not null identity, names varchar( ...
- Jmeter笔记2:参数化(五种方法)
案例:邮箱登录操作,参数化登录的用户名 方法一.依赖Jmeter自带的函数助手 选项-->函数助手对话框,即可打开函数助手弹窗 (1)比如使用函数_Random 输入最小值.最大值,点击下方的[ ...
- Recover deleted pictures in iOS 9
A case about business secret. The suspect is an engineer in Hitec company, and compeitiors pay lots ...