PTA-7-22 堆栈模拟队列

本题考点:采用堆栈模拟队列

目录

  • 解题思路
  • 情况分析
  • 代码实现

设已知有两个堆栈S1和S2,请用这两个堆栈模拟出一个队列Q。

所谓用堆栈模拟队列,实际上就是通过调用堆栈的下列操作函数:

  • int IsFull(Stack S):判断堆栈S是否已满,返回1或0;
  • int IsEmpty (Stack S ):判断堆栈S是否为空,返回1或0;
  • void Push(Stack S, ElementType item ):将元素item压入堆栈S
  • ElementType Pop(Stack S ):删除并返回S的栈顶元素。

实现队列的操作,即入队void AddQ(ElementType item)和出队ElementType DeleteQ()

输入格式:

输入首先给出两个正整数N1N2,表示堆栈S1S2的最大容量。随后给出一系列的队列操作:A item表示将item入列(这里假设item为整型数字);D表示出队操作;T表示输入结束。

输出格式:

对输入中的每个D操作,输出相应出队的数字,或者错误信息ERROR:Empty。如果入队操作无法执行,也需要输出ERROR:Full。每个输出占1行。

输入样例:

3 2
A 1 A 2 A 3 A 4 A 5 D A 6 D A 7 D A 8 D D D D T

输出样例:

ERROR:Full
1
ERROR:Full
2
3
4
7
8
ERROR:Empty

解题思路

采用堆栈模拟队列,堆栈是 FILO,而队列是 FIFO,所以如果我们需要用堆栈模拟队列,至少 需要两个栈。

第一个栈栈用来保存输入的值,而采用第二个栈栈来保存从较小的栈中移动过去的值,此时较大的栈顶部就是最先输入的值了。

而在本题中,所给的两个栈的大小不一样,所以需要选定一个栈当作输入栈,一个栈作为输出栈。

如果我们想尽可能保存多的值,我们需要将较小的栈作为输入栈,较大的栈作为输出栈,这是因为:如果把较大的栈作为输入栈的话,较小的栈无法缓冲从较大的栈中转移过来的值。

在输入过程中,一共会遇到这么几种情况(这里假定较小的栈为 s1,较大的栈为 s2):

情况分析

A(输入):

  1. s1 没满,直接放入 s1
  2. s1 满了,s2 为空,先把 s1 中的值都暂存到 s2 中,再保存到 s1
  3. s1 满了, s2 非空,此时需要输出 ERROR:Full (这是因为如果 s1 满了,无论再怎么放入 s1 或者 s2 都会打乱输出的顺序 )

D(输出):

4. s2 非空,直接输出 s2.top()

5. s1 非空, s2 为空, 那么将 s1 中的值放入 s2 中,然后输出 s2.top()

6. s1s2 都为空,那么输出 ERROR:Empty

代码实现

完整的代码实现如下:

/*
    Author: Veeupup
 */
#include <iostream>
#include <cstdio>
#include <cstdint>
#include <stack>
using namespace std;

int s1Size, s2Size;
stack<int> s1; // 容量较小,作为输入栈
stack<int> s2; // 容量较大,作为输出栈

int main()
{
    scanf("%d%d", &s1Size, &s2Size);
    getchar();
    if (s1Size > s2Size)
        swap(s1Size, s2Size); // 较小的为输入栈
    int s1Num = 0, s2Num = 0, tempNum;
    char next;
    while (scanf("%c", &next) != EOF)
    {
        if(next == ‘ ‘)
            continue;
        if (next == ‘A‘)
        {
            scanf("%d", &tempNum);
            if (s1Num < s1Size)
            {   // s1 不满,s2 为空,直接放入 s1 中
                s1.push(tempNum);
                s1Num++;
            }
            else if (s1Num == s1Size && s2Num == 0)
            {   // s1 满,s2 空,将 s1 中都放入 s2 中,再放入 s1 中
                while (!s1.empty())
                {
                    s2.push(s1.top());
                    s1.pop();
                    s2Num++;
                    s1Num--;
                }
                s1.push(tempNum);
                s1Num++;
            }
            else if (s1Num == s1Size && s2Num > 0)
            {   // s1 满,s2 不为空,输出错误
                printf("ERROR:Full\n");
            }
        }
        else if (next == ‘D‘)
        {
            if (s2Num > 0)
            {
                printf("%d\n", s2.top());
                s2.pop();
                s2Num--;
            }
            else if (s1Num > 0 && s2Num == 0)
            {
                while (!s1.empty())
                {
                    s2.push(s1.top());
                    s2Num++;
                    s1Num--;
                    s1.pop();
                }
                printf("%d\n", s2.top());
                s2Num--;
                s2.pop();
            }
            else if (s1Num == 0 && s2Num == 0)
            {
                printf("ERROR:Empty\n");
            }
        }
    }
    return 0;
}

原文地址:https://www.cnblogs.com/veeupup/p/12628869.html

时间: 2024-11-05 13:45:20

PTA-7-22 堆栈模拟队列的相关文章

3-08. 堆栈模拟队列(25)(ZJU_PAT 模拟)

题目链接:http://pat.zju.edu.cn/contests/ds/3-08 设已知有两个堆栈S1和S2,请用这两个堆栈模拟出一个队列Q. 所谓用堆栈模拟队列,实际上就是通过调用堆栈的下列操作函数: (1) int IsFull(Stack S):判断堆栈S是否已满,返回1或0: (2) int IsEmpty (Stack S ):判断堆栈S是否为空,返回1或0: (3) void Push(Stack S, ElementType item ):将元素item压入堆栈S: (4)

7-22 堆栈模拟队列

7-22 堆栈模拟队列(25 分) 设已知有两个堆栈S1和S2,请用这两个堆栈模拟出一个队列Q. 所谓用堆栈模拟队列,实际上就是通过调用堆栈的下列操作函数: int IsFull(Stack S):判断堆栈S是否已满,返回1或0: int IsEmpty (Stack S ):判断堆栈S是否为空,返回1或0: void Push(Stack S, ElementType item ):将元素item压入堆栈S: ElementType Pop(Stack S ):删除并返回S的栈顶元素. 实现队

7-22 堆栈模拟队列 (25分)

没注意看题,一开始把元素类型弄成char了,搞了好久都AC不了,换成int一次就AC了. 题意: 即用两个栈来模拟队列,使两个栈协作实现队列的功能. 思路: 1.第一个栈为输入栈,第二个栈为输出栈,输入栈比输出栈要小. 2.栈满条件:输入栈满了而输出栈不为空,说明栈满了,因为输出栈还有元素的话,输入栈的元素是不能搬到输出栈的,这样会造成顺序混乱(输出栈为空时连续多次将对个输入栈栈顶元素搬到输出栈这一情况除外). 3.所以说,输入栈的元素要搬到栈顶只能在一种条件下:就是输出栈为空时,且输入栈元素必

《模拟队列或堆栈》

1 package cn.itcast.api.b.list.subclass; 2 3 import java.util.LinkedList; 4 5 public class LinkedListTest { 6 7 public static void main(String[] args) { 8 /* 9 * 面试题:用LinkedList模拟一个堆栈或者队列数据结构. 10 * 创建一个堆栈或者队列数据结构对象.该对象中是使用LinkedList来完成的. 11 * 12 */ 1

【Java学习笔记】&lt;集合框架&gt;使用LinkedList来模拟一个堆栈或者队列的数据结构

1 import java.util.LinkedList; 2 3 public class Test5 { 4 5 public static void main(String[] args) { 6 7 Duilie dl = new Duilie(); 8 9 dl.myAdd("abc1"); 10 dl.myAdd("abc2"); 11 dl.myAdd("abc3"); 12 dl.myAdd("abc4");

Java集合框架之LinkedList-----用LinkedList模拟队列和堆栈

LinkedList的特有方法: (一)添加方法 addFisrt(E e):将指定元素插入此列表的开头.//参数e可以理解成Object对象,因为列表可以接收任何类型的对象,所以e就是Object对象(传递过程即向上转型). addLast(E e):将指定元素插入此列表的结尾. JDK1.6之后: offerFirst(); offerLast();//其实前后的用法相同,换了一个名字而已. (二):获取元素方法(获取过程不删除链表元素): getFirst();返回此列表的第一个元素.如果

用LinkedList模拟一个堆栈或者队列数据结构 总结: LinkedList知识点

/** 用LinkedList模拟一个堆栈或者队列数据结构. 创建一个堆栈和队列数据结构对象,该对象中使用LinkedList来完成的. 知识点总结: 1.LinkedList特点:链表式数据结构. 重复有序,查询速度慢,增删速度快.不同步的. 2.LinkedList除了实现List接口增删改查的功能外,有一些特有的方法,能够实现在List(列表)的开头和结尾 插入,删除,获取等特有功能.这就是为什么LinkedList能够模拟一个堆栈,或者队列,双端队列的数据结构了. 涉及知识点: 1.什么

使用LinkedList模拟一个堆栈或者队列数据结构

使用LinkedList模拟一个堆栈或者队列数据结构. 堆栈:先进后出  如同一个杯子. 队列:先进先出  如同一个水管. import java.util.LinkedList; public class DuiLie { private LinkedList link; public DuiLie() { link = new LinkedList(); } public void myAdd(Object obj) { link.addFirst(obj); } public Object

java集合 collection-list-LinkedList 模拟一个堆栈或者队列数据结构。

/* 使用LinkedList模拟一个堆栈或者队列数据结构. 堆栈:先进后出 如同一个杯子. 队列:先进先出 First in First out FIFO 如同一个水管. */ import java.util.*; class DuiLie { private LinkedList link; DuiLie() { link = new LinkedList(); } public void myAdd(Object obj) { link.addFirst(obj); } public O