Java实现栈和最小栈

一、栈的简介

栈(Stack)是一种常用的数据结构,其核心特点是先进后出。栈主要提供三种基本操作:push(入栈,将元素放入栈顶)、pop(出栈,取出并删除栈顶元素)、peek/top(查看栈顶元素但不删除)。栈可以用数组或链表实现,数组实现操作简单、随机访问快,但容量固定或需要扩容;链表实现则不受容量限制,但需要额外的指针空间。最小栈(MinStack)还可以在 O(1) 时间内获取当前栈的最小值,通过额外的辅助栈记录历史最小值实现。

二、IStack

复制代码
public interface IStack {
    void push(int x);
    int pop();
    int size();
    boolean empty();
    boolean full();
}

这段代码定义了一个栈的接口 IStack,用于规范栈的基本功能。接口中声明了五个方法:

push(int x):将元素 x 入栈。

pop():从栈顶弹出元素并返回,如果栈为空,通常会抛出异常。

size():返回栈中当前元素的个数。

empty():判断栈是否为空。

full():判断栈是否已满。

三、MyStack

复制代码
import java.util.Arrays;

public class MyStack implements IStack{

    private int[] elem;
    private int usedSize;
    private static final int DEFAULT_CAPACITY=10;
    public MyStack(){
        elem=new int[DEFAULT_CAPACITY];
    }

    @Override
    public void push(int x) {
        if(full()){
            elem=Arrays.copyOf(elem,2*elem.length);
        }
        elem[usedSize++]=x;

    }

    @Override
    public int pop() {
        if(empty()){
            throw new EmptyException("栈空了");
        }
        int old=elem[usedSize-1];
        usedSize--;
        return old;
    }

    public int peek(){
        if(empty()){
            throw new EmptyException("栈空了");
        }
        return elem[usedSize-1];
    }
    @Override
    public int size() {
        return usedSize;
    }

    @Override
    public boolean empty() {
        return usedSize==0;
    }

    @Override
    public boolean full() {
        if(usedSize==elem.length){
            return true;
        }
        return false;
    }
}

这段代码实现了一个顺序栈(数组栈),它通过数组 elem 存储栈中的元素,并用 usedSize 记录当前栈中元素的个数。栈的容量初始为 DEFAULT_CAPACITY(10),当数组满时,push 方法会通过 Arrays.copyOf 将数组扩容为原来的两倍,以保证栈可以动态增长。

push(int x) 将元素放入栈顶并更新 usedSize

pop() 从栈顶弹出元素并返回,同时判断栈是否为空,若为空则抛出自定义异常 EmptyException

peek() 查看栈顶元素但不删除,也会在空栈时抛异常。

size() 返回栈中当前元素数量,empty() 判断栈是否为空,full() 判断栈是否已满。

四、MinStack

复制代码
import java.util.Stack;

public class MinStack {
    private Stack<Integer> stack;
    private Stack<Integer> minStack;

    public MinStack(){
        stack=new Stack<>();
        minStack=new Stack<>();
    }

    public void push(int val){
        stack.push(val);
        if(minStack.empty()){
            minStack.push(val);
        }else{
            int peekVal=minStack.peek();
            if(val<peekVal){
                minStack.push(val);
            }
        }
    }

    public void pop(){
        int val=stack.pop();
        if(!minStack.empty()){
            if(val== minStack.peek()){
                minStack.pop();
            }
        }
    }

    public int top(){
        return stack.peek();
    }

    public int getMin(){
        if(!minStack.empty()){
            return minStack.peek();
        }
        return -1;
    }
}
import java.util.Stack;

public class MinStack {
    private Stack<Integer> stack;
    private Stack<Integer> minStack;

    public MinStack(){
        stack=new Stack<>();
        minStack=new Stack<>();
    }

    public void push(int val){
        stack.push(val);
        if(minStack.empty()){
            minStack.push(val);
        }else{
            int peekVal=minStack.peek();
            if(val<peekVal){
                minStack.push(val);
            }
        }
    }

    public void pop(){
        int val=stack.pop();
        if(!minStack.empty()){
            if(val== minStack.peek()){
                minStack.pop();
            }
        }
    }

    public int top(){
        return stack.peek();
    }

    public int getMin(){
        if(!minStack.empty()){
            return minStack.peek();
        }
        return -1;
    }
}

这段代码实现了一个最小栈,它可以在 O(1) 时间内获取当前栈中的最小值。代码使用了两个 Stack<Integer> 对象:stack 用于存储所有入栈的元素,而 minStack 用于记录栈中历史最小值。

当执行 push 操作时,如果 minStack 为空或新元素比当前最小值小,就将新元素压入 minStack;这样 minStack 的栈顶始终是当前最小值。

pop 操作会先从 stack 弹出元素,如果弹出的值等于 minStack 栈顶,也同步弹出 minStack 的栈顶,以保证最小值的正确性。

top() 方法返回当前栈顶元素,而 getMin() 返回当前最小值。

五、EmptyException

复制代码
public class EmptyException extends RuntimeException{
    public EmptyException(String msg){
        super(msg);
    }
}

这段代码定义了一个自定义异常类 EmptyException,它继承自 Java 的 RuntimeException,用于在程序运行时表示"栈或队列为空"的特殊情况。构造方法 EmptyException(String msg) 接收一个字符串参数 msg,并调用父类 RuntimeException 的构造方法,将提示信息传递给异常对象。当栈或队列等数据结构在执行 poppeek 等操作时,如果当前没有元素,就可以抛出这个异常,从而明确地告知调用者操作失败的原因。

相关推荐
划水的code搬运工小李2 分钟前
下载CSDN到PDF
开发语言·pdf·swift
不负岁月无痕6 分钟前
STL-- C++ stack_queue _priority_queue类 模拟实现
开发语言·c++
半个烧饼不加肉8 分钟前
JS 底层探究--上下文
开发语言·javascript·ecmascript
小满Autumn10 分钟前
依赖注入设计模式速查手册
开发语言·c#·wpf·mvvm·依赖注入
selt79115 分钟前
Redisson 源码深度分析
java·c++·redis·lua
装不满的克莱因瓶18 分钟前
Servlet 到 Spring MVC 架构演进:Java Web 开发二十年技术变迁史
java·spring·servlet·架构·springmvc
周末也要写八哥18 分钟前
浅谈:C++中cpp 14 ~ cpp 17
开发语言·c++·算法
不会C语言的男孩21 分钟前
C++ Primer 第13章:拷贝控制
开发语言·c++
z落落25 分钟前
C# 静态成员 vs 非静态成员(调用规则+内存特点)+只读和常量 const常量 / readonly / static readonly 三者终极区别
java·开发语言·c#
zhangfeng113327 分钟前
超算中心 高性能计算 slurm的linux版本 centos7,如何安装docker,如何安装torch2.4
linux·运维·服务器·开发语言·人工智能·机器学习·docker