news 2026/6/10 11:04:10

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

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
使用两个栈来实现一个队列

在 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)),实际开发中更常用。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/10 13:20:55

告别听不清困境,声网STT让每一次沟通都被精准捕捉

常年跑项目、随时对接客户&#xff0c;移动办公是我的日常&#xff0c;却被 STT 工具坑怕了&#xff01;咖啡厅开会&#xff0c;邻桌嘈杂让转录内容杂乱&#xff0c;客户需求误判&#xff0c;会后返工一下午&#xff1b;地铁接电话&#xff0c;列车轰鸣和报站声干扰&#xff0c…

作者头像 李华
网站建设 2026/6/9 18:33:15

上海比较好的港澳台联考学校哪家专业

上海比较好的港澳台联考学校哪家专业在上海&#xff0c;对于众多港澳台学生而言&#xff0c;选择一所专业的港澳台联考学校至关重要。这不仅关系到他们的学业发展&#xff0c;更影响着未来的升学与职业规划。那么&#xff0c;上海比较好的港澳台联考学校哪家专业呢&#xff1f;…

作者头像 李华
网站建设 2026/6/10 13:19:37

天然与责任的平衡:为什么选择 mixomi 海豹油?

随着健康意识的不断提升&#xff0c;消费者对于营养补充品的选择变得越来越理性。现代消费者不仅关注效果&#xff0c;更注重产品的来源、工艺和品牌的社会责任。尤其是在 Omega-3 脂肪酸的补充上&#xff0c;市场上有许多选择&#xff0c;例如鱼油、藻油等&#xff0c;但如何找…

作者头像 李华
网站建设 2026/6/10 14:06:37

YOLOv8 Pose姿态估计功能实战演示

YOLOv8 Pose姿态估计功能实战解析 在智能制造车间里&#xff0c;一台边缘计算盒子正实时监控着流水线工人的操作动作。当某位员工弯腰搬运重物的姿势不符合安全规范时&#xff0c;系统立刻发出语音提醒——这一切的背后&#xff0c;正是 YOLOv8 Pose 在默默发挥作用。它不仅识别…

作者头像 李华