用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 数据结构与算法之美 - 线性表(数组、栈、队列、链表)
前言 基础知识就像是一座大楼的地基,它决定了我们的技术高度. 我们应该多掌握一些可移值的技术或者再过十几年应该都不会过时的技术,数据结构与算法就是其中之一. 栈.队列.链表.堆 是数据结构与算法中的基 ...
随机推荐
- XBox360-双光盘游戏自制GOD
一直在找极限竞速4(Forza4),虽然这个版本比较老,但因为带体感.终于下到了,可惜是2个ISO.试着自己做GOD.用到两个软件:Iso2God和Xbox Backup Creator(俗称XBC) ...
- 问题:glGenBuffers()函数没有定义怎么办
链接glew.lib库,#include <gl/glew.h>. glew是opengl 的扩展库
- Html类ImageGetter接口
public class ImgLabelActivity extends Activity { private static final String TAG = "ImgLabelAct ...
- 【caffe-windows】 caffe-master 之 classfication_demo.m 超详细分析
classification_demo.m 是个很好的学习资料,了解这个代码之后,就能在matlab里用训练好的model对输入图像进行分类了,而且我在里边还学到了oversample的实例,终于了解 ...
- 使用 Environment Indicator 模块区分不同的 Drupal 环境
每个 Drupal 网站建设人员到了某个时期,都会有误将线上站点当做本地站点进行修改的经历.尤其是在浏览器中打开了几十个页面时,很容易忘记究竟哪个是哪个. Environment Indicator ...
- <关于数据仓库>基于docker的Mysql与Hadoop/Hive之间的数据转移 (使用Apache Sqoop™)
原创博客,转载请联系博主! 摘要:本文介绍了如何使用docker快速搭建一个可以从外部访问的mysql服务容器,和由docker搭建的分布式Hadoop文件系统,并且使用ApacheSqoop完成将m ...
- number对象,bom对象
number对象 新创建一个number的对象,toFixed是精确到位数 var num =new Number('123.1231'); console.log(num.toFixed(1)); ...
- c# 将页面导出到word(含图片及控件)
/// <summary> /// 创建word /// <param name="filePath">文件路径 </param> /// &l ...
- javascript设计模式-工厂模式
简单工厂模式:使用一个类来生成实例. 复杂工厂模式:使用子类来决定一个成员变量应该是哪个具体的类的实例. 简单工厂模式是由一个方法来决定到底要创建哪个类的实例, 而这些实例经常都拥有相同的接口.通过工 ...
- js,jquery获取下拉框选中的option
js获取select选中的值: var sel=document.getElementById("select1"); var index = sel.selectedIndex; ...