不得不说编程之美是一本好书,虽然很多题目在做acm中的过程中遇到过,不过还是有很多值得思考的地方

这是今天在编程之美上看到的一个问题,对于栈转化成队列的一个思考

平时都太过依赖c++内函数库中的栈和队列,但是对于他们的扩展我们还是应该自己进行手写栈和队列来实现更简单的算法

题目大意:

假设有这样一个拥有3个操作的队列:

1. EnQueue(v) : 将 v 加入队列

2. DeQueue: 使队列中队首元素删除并返回此元素

3.MaxElement: 返回队列中的最大元素

设计一种数据结构和算法,让MaxElement操作的时间复杂度尽可能地低

首先来思考这样一个问题:

对于前面已知的一段序列,我们加入一个元素x,得到当前序列的最大值会特别容易

maxVal[i] = max(maxVal[i-1] , x);

那么我们删掉第 i 个元素,得到的序列中的最大值也特别容易就是maxVal[i-1]

换个角度想想此时的第 i 个元素可理解为栈顶元素

也就是说这道题目改成拥有3个栈操作的队列那么我们直接就可以在单位时间内得到MaxElement

而保存这些数也就是把所有数扫一遍,O(N)就解决了

但是这里是队列,队列总是删去最前面也就是下标为 0 的元素,这样我们无法利用maxVal[i]来表示当前队列元素中的最大值

那我们可以考虑,既然栈如此方便,如何将队列用栈来表示

首先我们需要两个栈,把数据放入一个栈s2,那么这些数据其实是倒着保存的 , 那么我们要删去队列的首元素,也就是删除栈底的元素,

那么我们可以利用另一个栈s1,将前一个栈s2中的元素再倒一遍进入另一个栈s1,那么此时s1的栈顶也就是队列的栈顶

也就是说只要s1中有元素,那么s1中的栈顶始终是队列首元素

只有当s1中元素都不存在时,才将s2中的元素一次性全倒入s1

int DeQueue()
    {
        if(s1.empty() && s2.empty()) return -INF; //队列中值不存在
        if(s1.empty()){
            while(!s2.empty()){
                s1.push(s2.pop());
            }
        }
        return s1.pop();
    }

那么队列中的所有元素总是分布在s1和s2中,那么最大元素既可能在s1,也可能在s2中,所以返回 return max(s1.Max() , s2.Max());

这样这个程序总的复杂度就能够达到O(N)的线性复杂度了

接下来是自己手写的一个测试过的没有问题的程序

 #include <cstdio>
#include <cstring>
#include <exception>
using namespace std;
#define max(a,b) a>b?a:b
const int MAXN = ;
const int INF = 0x3fffffff;
struct Stack{
int top , a[MAXN] , maxVal[MAXN];
//a[]保存栈中的元素,maxVal[i]保存i以及前方的最大值,top为栈顶指针,初始为-1
Stack(){
top = -;
memset(maxVal , - , sizeof(maxVal));
}
void push(int x)
{
if(top>=MAXN) throw exception();
a[++top] = x;
if(top == ) maxVal[top]=x;
else maxVal[top] = max(maxVal[top-] , x);
}
bool empty(){return top == -;}
int Max()
{
if(empty()) return -INF;
return maxVal[top];
}
int pop()
{
if(top<) return -;
int ret = a[top--];
return ret;
}
}; struct Queue{
Stack s1 , s2;
Queue()
{
s1 = Stack();
s2 = Stack();
}
void EnQueue(int x)
{
s2.push(x);
}
int DeQueue()
{
if(s1.empty() && s2.empty()) return -INF; //队列中值不存在
if(s1.empty()){
while(!s2.empty()){
s1.push(s2.pop());
}
}
return s1.pop();
}
int MaxElement()
{
return max(s1.Max() , s2.Max());
}
bool empty()
{
return s1.empty()&&s2.empty();
}
}; int main()
{
// freopen("a.in" , "r" , stdin);
Queue q; q = Queue();
// int num[10] = {5 , 7 , 6 , 3 , 2 , 4 , 9 , 15 , 13 , 11};
q.EnQueue();
int op , x;
while(!q.empty())
{
printf("输入操作类型:");
scanf("%d" , &op);
if(op == ){
printf("输入一个数入队列:");
scanf("%d" , &x);
q.EnQueue(x);
}
else if(op == ){
printf("队列首元素 %d 从队列中退出\n" , q.DeQueue());
}
else{
printf("当前队列最大元素为 %d\n" , q.MaxElement());
} //下方每次用来检测队列中两个栈的数据保存情况
/*
printf("输出栈1中的元素\n");
for(int i=q.s1.top ; i>=0 ; i--){
printf("%d " , q.s1.a[i]);
}
puts(""); printf("输出栈2中的元素\n");
for(int i=q.s2.top ; i>=0 ; i--){
printf("%d " , q.s2.a[i]);
}
puts("");
*/
}
return ;
}

<<编程之美>> -- 队列中取最大值操作的问题的更多相关文章

  1. 编程之美 set 10 队列中取最大值操作问题

    题目 假设有这样一个拥有三个操作的队列 1. Enqueue(v) 2. Dequeue() 3. MaxEle() 请设计一种数据结构和算法, 让 MAXELE 操作的时间复杂度尽可能的低 思路 1 ...

  2. 在含有null值的复杂类的集合(Collection)中取最大值

    在日常编程中,经常遇到要在一组复杂类的集合(Collection)中做比较.取最大值或最小值. 举个最简单的例子,我们要在一个如下结构的集合中选取包含最大值的元素: public class Clas ...

  3. Java并发编程:阻塞队列(转载)

    Java并发编程:阻塞队列 在前面几篇文章中,我们讨论了同步容器(Hashtable.Vector),也讨论了并发容器(ConcurrentHashMap.CopyOnWriteArrayList), ...

  4. 【转】Java并发编程:阻塞队列

    在前面几篇文章中,我们讨论了同步容器(Hashtable.Vector),也讨论了并发容器(ConcurrentHashMap.CopyOnWriteArrayList),这些工具都为我们编写多线程程 ...

  5. Java并发编程:阻塞队列 <转>

    在前面几篇文章中,我们讨论了同步容器(Hashtable.Vector),也讨论了并发容器(ConcurrentHashMap.CopyOnWriteArrayList),这些工具都为我们编写多线程程 ...

  6. 12、Java并发编程:阻塞队列

    Java并发编程:阻塞队列 在前面几篇文章中,我们讨论了同步容器(Hashtable.Vector),也讨论了并发容器(ConcurrentHashMap.CopyOnWriteArrayList), ...

  7. (转)Java并发编程:阻塞队列

    Java并发编程:阻塞队列 在前面几篇文章中,我们讨论了同步容器(Hashtable.Vector),也讨论了并发容器(ConcurrentHashMap.CopyOnWriteArrayList), ...

  8. linux编程之消息队列

    消息队列是内核地址空间中的内部链表,通过linux内核在各个进程之间传递内容,消息顺序地发送到消息队列中,并且以几种不同的方式 从队列中获取,每个消息队列可以用IPC标识符唯一的进行标识,内核中的消息 ...

  9. JUC 并发编程--06, 阻塞队列(7种), 阻塞等待 api的 代码验证

    这些队列的 api ,就是添加队列,出队列,检测对首元素, 由于 add()--remove(), offer()--poll(),太简单这里不做验证, 只验证后二组api: 阻塞等待( put()- ...

随机推荐

  1. 对数组随机赋值,并输出(Arrays.toString(arr))

    import java.util.Arrays; public class Demo { public static void main(String[] args) { int[] arr = ne ...

  2. css超出部分显示省略号

    单行文本 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF- ...

  3. glassfish应用服务器安装配置

    1.Glassfish4.0下载地址:https://glassfish.java.net/download.html#gfoseTab 2.将下载的glassfish-4.0.zip传输到服务器/h ...

  4. tuple元组创建单元素

    创建tuple单元素,一定要在结尾时添加一个逗号(,)解:如果不加逗号,哪怕是使用tuple()正确的创建元组,也会有歧义,它会把创建tuple元组的单元素,当成一个普通的输出语句结果列:如下,错误的 ...

  5. 自定义button上传按钮

    <div class="upload_files"> <input type="file" class="upload_icon&q ...

  6. shutil模块 + shelve模块 二合一版

    其他的看我前面的博客 import shutil # 将文件内容拷贝到另一个文件with open('old.xml','r') as read_f,open('new.xml', 'w') as w ...

  7. 04全志R58平台编译内核需要选择的配置

    04全志R58平台编译内核需要选择的配置 2018/11/6 14:19 版本:V1.0 开发板:SC5806 1.系统编译:(略) 每次系统编译/内核的时候都需要选3次N: * * Xtables ...

  8. 关于Android软键盘把布局顶上去的问题(一)

    最近接触到了一个登陆页面,布局最上面显示的是一个波纹的view,中间显示账号和密码的EditText,紧接着还有一个Button: 希望:点击EditText时,软键盘不能把波纹的view顶出去,也不 ...

  9. Jvisualvm--JAVA性能分析工具

    JDK自带的JAVA性能分析工具.它已经在你的JDK bin目录里了,只要你使用的是JDK1.6 Update7之后的版本.点击一下jvisualvm.exe图标它就可以运行了. 这里是VisualV ...

  10. win7 快捷键 收集

    1. 轻松访问键盘快捷方式 按住右 Shift 八秒钟:启用和关闭筛选键 按左 Alt + 左 Shift + PrtScn (或 PrtScn):启用或关闭高对比度 按左 Alt + 左 Shift ...