用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 数据结构与算法之美 - 线性表(数组、栈、队列、链表)
前言 基础知识就像是一座大楼的地基,它决定了我们的技术高度. 我们应该多掌握一些可移值的技术或者再过十几年应该都不会过时的技术,数据结构与算法就是其中之一. 栈.队列.链表.堆 是数据结构与算法中的基 ...
随机推荐
- The 2013 ACM-ICPC Asia Changsha Regional Contest - K
Pocket Cube Time Limit: 2 Seconds Memory Limit: 65536 KB Pocket Cube is a 3-D combination puzzl ...
- The Ninth Hunan Collegiate Programming Contest (2013) Problem I
Problem I Interesting Calculator There is an interesting calculator. It has 3 rows of button. Row 1: ...
- VC让对话框显示就最大化
方法一:在OnInitDialog()函数中 ShowWindow(SW_SHOWMAXIMIZED); 初始化的时候 方法二: 当然,你可以获取屏幕大小,然后设置窗口位置/大小 //ShowWind ...
- 链表:删除链表中重复的结点(java实现)
题目描述 在一个排序的链表中,存在重复的结点,请删除该链表中重复的结点,重复的结点不保留,返回链表头指针. 例如,链表1->2->3->3->4->4->5 处理后 ...
- java解析xml的4种经典方法
========================================== xml文件 <?xml version="1.0" encoding="GB2 ...
- linux下启动dbca或netmgr类的图形界面报错解决
linux下启动dbca或netmgr类的图形界面报错解决 Xlib: connection to ":0.0" refused by server Xlib: No pro ...
- NOIP2013Day1解题报告
本来今天晚上拿13年NOIP的题目来做一下,测测能够得多少分,结果一晚上把Day1写完竟然AK了,吼吼吼 D1T1,题目:http://codevs.cn/problem/3285/ 很水的一道快速幂 ...
- [工具] 如何利用Notepad++去除重复行
问题: 需要去除重复数据, 例如: 解决方案: 1. 打开notepad++: 2. 如果没有找到"TextFx" 选项, 需要先安装该插件. 依次打开"插件" ...
- 【Flex教程】#009 As/typeof /instanceof /is的作用
“as” :主要用它做类型转化 假设有一个类叫做Class1,我们声明了一个它的对象 c1,如果想要将它转换成Class2类型,只要这样写: Class2(c1); AS3 中的操作符: as 实现就 ...
- MSP430F149学习之路——蓝牙模块
注意蓝牙模块的接法! #include <msp430x14x.h> ]; ; void int_clk() { BCSCTL1 &= ~XT2OFF; BCSCTL2 |= SE ...