当前位置: 首页 > news >正文

使用两个栈来实现一个队列

在 Java 中,利用两个栈实现队列的核心思路是通过栈的“后进先出”特性模拟队列的“先进先出”特性:将一个栈(stackIn)作为“入队栈”,专门处理入队操作;另一个栈(stackOut)作为“出队栈”,专门处理出队/获取队首操作。只有当stackOut为空时,才将stackIn的所有元素转移到stackOut,从而实现“先进先出”。

核心原理

  1. 入队(offer):直接将元素压入stackIn(O(1) 时间复杂度)。
  2. 出队(poll)
    • stackOut为空,将stackIn中所有元素依次弹出并压入stackOut(此时stackOut的栈顶就是队列的队首元素);
    • 弹出stackOut的栈顶元素(即队列的队首元素)。
  1. 获取队首(peek):逻辑与poll类似,但仅获取stackOut的栈顶元素,不弹出。
  2. 判空(isEmpty):当stackInstackOut都为空时,队列为空。

完整代码实现

import java.util.Stack; /** * 用两个栈实现队列 */ public class QueueByTwoStacks { // 入队栈:专门处理入队操作 private Stack<Integer> stackIn; // 出队栈:专门处理出队/获取队首操作 private Stack<Integer> stackOut; // 初始化 public QueueByTwoStacks() { stackIn = new Stack<>(); stackOut = new Stack<>(); } /** * 入队操作:直接压入入队栈 * @param x 要入队的元素 */ public void offer(int x) { stackIn.push(x); } /** * 出队操作:弹出队首元素 * @return 队首元素 * @throws RuntimeException 队列为空时抛出异常 */ public int poll() { // 先检查出队栈是否为空,为空则转移入队栈的元素 transferIfNeeded(); // 若仍为空,说明队列为空 if (stackOut.isEmpty()) { throw new RuntimeException("队列为空,无法执行poll操作"); } // 弹出出队栈的栈顶(队列的队首) return stackOut.pop(); } /** * 获取队首元素(不弹出) * @return 队首元素 * @throws RuntimeException 队列为空时抛出异常 */ public int peek() { // 先检查出队栈是否为空,为空则转移入队栈的元素 transferIfNeeded(); // 若仍为空,说明队列为空 if (stackOut.isEmpty()) { throw new RuntimeException("队列为空,无法执行peek操作"); } // 获取出队栈的栈顶(队列的队首) return stackOut.peek(); } /** * 判断队列是否为空 * @return 空返回true,否则返回false */ public boolean isEmpty() { return stackIn.isEmpty() && stackOut.isEmpty(); } /** * 辅助方法:当出队栈为空时,将入队栈的所有元素转移到出队栈 */ private void transferIfNeeded() { if (stackOut.isEmpty()) { while (!stackIn.isEmpty()) { stackOut.push(stackIn.pop()); } } } // 测试示例 public static void main(String[] args) { QueueByTwoStacks queue = new QueueByTwoStacks(); // 入队:1 -> 2 -> 3 queue.offer(1); queue.offer(2); queue.offer(3); // 输出队首:1 System.out.println("队首元素:" + queue.peek()); // 出队:1 System.out.println("出队元素:" + queue.poll()); // 输出队首:2 System.out.println("队首元素:" + queue.peek()); // 出队:2 System.out.println("出队元素:" + queue.poll()); // 入队:4 queue.offer(4); // 出队:3 System.out.println("出队元素:" + queue.poll()); // 出队:4 System.out.println("出队元素:" + queue.poll()); // 判空:true System.out.println("队列是否为空:" + queue.isEmpty()); } }

代码说明

  1. 栈的选择:直接使用 Java 内置的Stack类(也可使用Deque接口的LinkedList实现,Deque是栈的推荐替代方案,Stack是旧版类)。
  2. 核心辅助方法transferIfNeeded()封装了“入队栈转出队栈”的逻辑,避免代码冗余,仅在stackOut为空时触发转移,保证效率。
  3. 异常处理pollpeek操作时,若队列为空则抛出运行时异常,符合队列的常规行为。
  4. 判空逻辑:必须同时判断stackInstackOut,因为可能存在“入队栈有元素但出队栈为空”的中间状态。

时间复杂度

  • offer(入队):O(1)(直接压栈)。
  • poll/peek(出队/获取队首)
    • 均摊复杂度 O(1):每个元素最多被“压入stackIn→ 弹出stackIn→ 压入stackOut→ 弹出stackOut”四次,分摊到每个操作上为 O(1);
    • 最坏复杂度 O(n):当stackOut为空时,需要转移n个元素(n为入队栈的元素数)。

关键总结

  • 双栈实现队列的核心是“分栈职责 + 按需转移”:入队只操作stackIn,出队只操作stackOut,仅当stackOut为空时才转移元素,避免重复转移。
  • 与双队列实现栈相比,双栈实现队列的均摊时间复杂度更优(pop/peek 为均摊 O(1)),实际开发中更常用。
http://www.jsqmd.com/news/99884/

相关文章:

  • MIL-STD-1553B总线仿真应用解析
  • Conda-forge构建SD3.5 FP8推理环境的正确姿势
  • 32、Linux系统磁盘管理与打印操作全解析
  • 2026中专直播电商,考什么证书找工作有优势?
  • 零基础部署Wan2.2-T2V-A14B:本地化视频生成全指南
  • 45、Linux系统使用指南:文件、多媒体与网络操作全解析
  • Ascend C高性能LayerNorm融合算子开发实战
  • 35、Linux实用技巧:日程管理、联系人管理与数学计算
  • EmotiVoice社区版与商业版功能对比指南
  • 开发者必看:LobeChat源码结构与二次开发入门路径
  • 告别听不清困境,声网STT让每一次沟通都被精准捕捉
  • [特殊字符] 如何让自定义音量条生效?彻底解决“按音量键只显示系统默认音量条”的问题
  • GitHub项目实践:Fork并定制你的个性化Anything-LLM前端界面
  • Fifth Assignment——Alpha Sprint
  • PaddlePaddle在企业级AI应用中的优势分析:开发便捷性与模型丰富性
  • IP地址信息查询API合集
  • YOLOv8 Pose姿态估计功能实战演示
  • BioSIM抗人TNFSF2/TNFα抗体SIM0348:专业品质与品牌保障
  • CodeSys执行G代码的CNC功能
  • 机房预约系统
  • PCB打板是否需要SMT贴片?——从工程实战角度看清本质
  • Docker安装TensorRT并暴露gRPC接口供外部调用
  • 2025 国际考生雅思报班指南:三大高认可度机构核心解析与选课策略 - 品牌测评鉴赏家
  • 42、互联网聊天与Linux系统管理全攻略
  • Win10下Anaconda配置TensorFlow-GPU 2.5.0完整指南
  • 2025年十大专业文创旅游规划品牌公司推荐,实力企业全解析 - mypinpai
  • 企业级AI客服系统搭建首选——LobeChat镜像全面解读
  • 清华镜像站同步频率揭秘:TensorFlow更新多久能同步?
  • 43、Linux系统使用与管理全解析
  • 2025煤质分析仪器TOP5权威推荐:闪点测定仪认证厂家,甄 - 工业品牌热点