最近刷剑指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. [VB.NET]Dictionary类

    字典类是一个很重要的类,尤其是对于数据的简单存储,查询,和处理. 废话不多说,简单记录下我探索的结果. 1. Dictionary内部索引是0基的.也就是说第一个元素的序号是0. 2. Public ...

  2. http是什么?

    http HyperText Transfer Protocol 超文本传输协议,是一个应用层通信协议. 可以用wireshark抓取.

  3. 如何用C代码生成二维码

    如何用C代码生成二维码 当下因微信和支付宝等手机应用广泛使用,而基于二维码/一维条码的移动支付,也借助手机移动端席卷全国,使得越来越多的人知道有"二维码"这么一种东西. 对于普通用 ...

  4. pb中读取大文本数据

    string ls_FileName,lb_FileDatas,lb_FileData  long ll_FileLen,ll_Handle,ll_Loop,ll_Bytes,ll_Loops,ll_ ...

  5. Duilib学习笔记《02》— 界面布局

    1. 界面描述XML文件 Duilib主要是通过XML来进行界面的布局配置,程序通过读取并解析XML文件来创建对应的窗体.DuiLib的页面布局分为三类:窗体(Window).容器(Contain)和 ...

  6. sql语句小练习一

    create database aaa go use aaa go create table student(    sno varchar(3), sname varchar(4) not null ...

  7. 高效渲染css

    译自:http://css-tricks.com/efficiently-rendering-css/ 无可否认我并不经常考虑这个问题… 我们书写的CSS的效率如何,浏览器渲染它的速度如何? 浏览器的 ...

  8. (笔记)angular 事件传递获取当前

  9. CopyOnWriteArrayList

    CopyOnWriteArrayList, 内部通过锁+volatile修饰的数组实现的 是一种线程安全的ArrayList,写操作时会copy一个新的内部数组出来替换掉旧 的数组.遍历操作不用加锁了 ...

  10. Oracle sqlldr使用示例

    SQLLDR用于快速的批量导入数据, 示例步骤如下: 1.确定Oracle的连接示例 username/password@sid 我使用的是:system/world @ localorcl sid是 ...