最近刷剑指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)的时间复杂度,找到栈和队列中的最小(大)值的更多相关文章

  1. 剑指offer:按之字形打印二叉树(栈|双向队列+中序遍历)

    1. 题目描述 /** 请实现一个函数按照之字形打印二叉树, 即第一行按照从左到右的顺序打印, 第二层按照从右至左的顺序打印, 第三行按照从左到右的顺序打印, 其他行以此类推. */ 2. 双向队列 ...

  2. 包含min函数的栈、队列

    题目:定义栈的数据结构,请在该类型中实现一个能够得到栈/队列的最小元素的min函数.在该栈/队列中,调用min.入栈(入队列)及出栈(出队列)函数的时间复杂度都是O(1). 1. 包含min函数的栈 ...

  3. Java的栈和队列

    package com.ipmotor.sm.db;import java.util.LinkedList;import java.util.Queue;import java.util.Stack; ...

  4. O(1)时间复杂度求栈中最小元素

    import java.util.Stack; /** * 功能:O(1)时间复杂度求栈中最小元素 * 思路:空间换取时间,使用两个栈,stack1栈存储数据,stack2栈存储最小值: * stac ...

  5. 如何在O(1)时间复杂度获取栈中最大值和最小值

    问题描述: 如何在O(1)时间复杂度获取栈中的最大值和最小值? 问题分析: 普通栈规定的push(入栈).pop(出栈).peek(查看栈顶)等操作都只能在栈顶上操作,如果栈中元素是有序的,那么我们就 ...

  6. 单调栈&单调队列入门

    单调队列是什么呢?可以直接从问题开始来展开. Poj 2823 给定一个数列,从左至右输出每个长度为m的数列段内的最小数和最大数. 数列长度:\(N <=10^6 ,m<=N\) 解法① ...

  7. java——栈和队列 面试题

    (1)栈的创建 (2)队列的创建 (3)两个栈实现一个队列 (4)两个队列实现一个栈 (5)设计含最小函数min()的栈,要求min.push.pop.的时间复杂度都是O(1) (6)判断栈的push ...

  8. 笔试算法题(05):转换BST为双向链表 & 查找栈中的最小元素

    出题:把二元查找树转变成排序的双向链表.输入一棵二元查找树,要求将该二元查找树按照中序转换成一个排序的双向链表,要求不能创建任何新的节点,只能调整指针的指向: 分析: 递归的思路,当前节点需要进行的处 ...

  9. JavaScript 数据结构与算法之美 - 线性表(数组、栈、队列、链表)

    前言 基础知识就像是一座大楼的地基,它决定了我们的技术高度. 我们应该多掌握一些可移值的技术或者再过十几年应该都不会过时的技术,数据结构与算法就是其中之一. 栈.队列.链表.堆 是数据结构与算法中的基 ...

随机推荐

  1. java——操作文件

    Java文件操作,共实现了文件复制(单个文件和多层目录文件),文件移动(单个文件和多层目录文件),文件删除(单个文件和多层目录文件),文件压缩 (单个文件),文件解压(单个文件),文件分割(将一个大文 ...

  2. SICP 1.1-1.5

    1.1 a = b = nil 1.2 (/ (+ (- (- (+ (/ ))))) (* (- ) (- )) 1.3 a = b = nil 1.4... 1.5 (define (p) (p) ...

  3. flex的Cairngorm框架

    由于要写flex的项目,接触了一段时间的Cairngorm框架,初步认识它是flex的一个mvc结构的框架实现了页面,调用相应方法的控制,和后台交互之间的三层之间的联系.Cairngorm框架主要包括 ...

  4. redis学习(4)redis安装部署

    下载redis-1.2.6.tar.gz 将下载包拷贝到/usr/local/webserver/redis-1.2.6/下 2.安装 tar -zxvf redis-1.2.6.tar.gz ce ...

  5. Rational Functional Tester 对象文件(rftdef)合并

    Rational Functional Tester (RFT) 是一款集成在Eclipse上的自动化测试工具.而自动化测试最麻烦的维护部分,一大部分也是指的是对象库文件的管理维护. 之前的项目里,经 ...

  6. OSGI.NET 插件无法启动之情景一

    关于osgi.net 的使用网上也有不少的资料,最近在使用osgi.net  开发插件的时候,遇到了这样的bug,造成插件甚至整个项目都无法启动,异常的具体消息如下: Could not find a ...

  7. js时间冒泡,阻止事件冒泡

    首先解释一下事件冒泡神什么, 在js中,假如在div中嵌套一个div 如 <style type="text/css"> #box1{width:500px;heigh ...

  8. php和apache安装心得

    又一次安装apache和php~~ 又一次心碎的体验~ 本想参考之前的apache配置之间进行安装(在windows 下编辑,然后传到linux上),怎耐,一直有问题,打开浏览器直接进行下载.发现是浏 ...

  9. 可视化数据包分析工具-CapAnalysis

    可视化数据包分析工具-CapAnalysis 我们知道,Xplico是一个从pcap文件中解析出IP流量数据的工具,本文介绍又一款实用工具-CapAnalysis(可视化数据包分析工具),将比Xpli ...

  10. jqmobi 的一些設置

    jqmobi version=2.1; 不是 version =3.0: 好吧,我用了jqmobi 差不多半年了,我竟然連 官方的文檔都沒有看完,怪不得我走了多少的彎路.....哎!!!! 1.隱藏 ...