Java代码实现顺序栈和链式栈

简介: 栈(stack)又名堆栈,它是一种运算受限的线性表。其限制是仅允许在表的一端进行插入或者删除运算。后进先出(Last In First Out)。栈中的数据操作主要有push(压入)和pop(弹出)操作。实际上,栈就可以用数组来实现,也可以用链表来实现。用数组实现的叫做顺序栈,用链表实现的叫做链式栈。为了简单起见,我们的栈不支持泛型。

Java代码实现顺序栈和链式栈

栈(stack)又名堆栈,它是一种运算受限的线性表。其限制是仅允许在表的一端进行插入或者删除运算。后进先出(Last In First Out)。

栈中的数据操作主要有push(压入)和pop(弹出)操作。

实际上,栈就可以用数组来实现,也可以用链表来实现。用数组实现的叫做顺序栈,用链表实现的叫做链式栈。

为了简单起见,我们的栈不支持泛型。

顺序栈

public class StackBasedArray implements StackInterface{

    private String[] values; // 存储栈中元素的数组
    private int capacity;// 栈的容量
    private int count;// 栈中已有数据个数

    // 初始化栈,容量确定。
    public StackBasedArray(int capacity) {
        this.values = new String[capacity];
        this.capacity = capacity;
        this.count = 0;
    }

    /**
     * 入栈操作
     *
     * @param value
     * @return
     */
    public Boolean push(String value) {
        // 数组空间不够了
        if (capacity == count) {
            return false;
        }
        values[count] = value;
        count++;
        return true;
    }

    /**
     * 出栈操作
     *
     * @return
     */
    public String pop() {
        // 栈为空,则直接返回null
        if (0 == count) {
            return null;
        }
        String tmp = values[count - 1];
        --count;
        return tmp;
    }

    @Override
    public String toString() {
        return "StackBasedArray{" +
                "values=" + Arrays.toString(values) +
                ", capacity=" + capacity +
                ", count=" + count +
                '}';
    }
}

测试代码:

        StackBasedArray as = new StackBasedArray(10);
        System.out.println(as);

        Boolean r1 = as.push("000");
        System.out.println(as + ",r1:" + r1);

        for (int i = 1; i < 11; i++) {
            boolean r2 = as.push("" + i + i + i);
            System.out.println(as + ",r2:" + r2);
        }

        System.out.println("as.pop():" + as.pop() + "," + as);
        System.out.println("as.pop():" + as.pop() + "," + as);
        System.out.println("as.pop():" + as.pop() + "," + as);
        System.out.println("as.pop():" + as.pop() + "," + as);
        System.out.println("as.pop():" + as.pop() + "," + as);
        System.out.println("as.pop():" + as.pop() + "," + as);
        System.out.println("as.pop():" + as.pop() + "," + as);
        System.out.println("as.pop():" + as.pop() + "," + as);
        System.out.println("as.pop():" + as.pop() + "," + as);
        System.out.println("as.pop():" + as.pop() + "," + as);
        System.out.println("as.pop():" + as.pop() + "," + as);
        System.out.println("as.pop():" + as.pop() + "," + as);

输出结果:符合预期

StackBasedArray{values=[null, null, null, null, null, null, null, null, null, null], capacity=10, count=0}
StackBasedArray{values=[000, null, null, null, null, null, null, null, null, null], capacity=10, count=1},r1:true
StackBasedArray{values=[000, 111, null, null, null, null, null, null, null, null], capacity=10, count=2},r2:true
StackBasedArray{values=[000, 111, 222, null, null, null, null, null, null, null], capacity=10, count=3},r2:true
StackBasedArray{values=[000, 111, 222, 333, null, null, null, null, null, null], capacity=10, count=4},r2:true
StackBasedArray{values=[000, 111, 222, 333, 444, null, null, null, null, null], capacity=10, count=5},r2:true
StackBasedArray{values=[000, 111, 222, 333, 444, 555, null, null, null, null], capacity=10, count=6},r2:true
StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, null, null, null], capacity=10, count=7},r2:true
StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, null, null], capacity=10, count=8},r2:true
StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, null], capacity=10, count=9},r2:true
StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, 999], capacity=10, count=10},r2:true
StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, 999], capacity=10, count=10},r2:false
as.pop():999,StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, 999], capacity=10, count=9}
as.pop():888,StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, 999], capacity=10, count=8}
as.pop():777,StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, 999], capacity=10, count=7}
as.pop():666,StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, 999], capacity=10, count=6}
as.pop():555,StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, 999], capacity=10, count=5}
as.pop():444,StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, 999], capacity=10, count=4}
as.pop():333,StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, 999], capacity=10, count=3}
as.pop():222,StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, 999], capacity=10, count=2}
as.pop():111,StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, 999], capacity=10, count=1}
as.pop():000,StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, 999], capacity=10, count=0}
as.pop():null,StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, 999], capacity=10, count=0}
as.pop():null,StackBasedArray{values=[000, 111, 222, 333, 444, 555, 666, 777, 888, 999], capacity=10, count=0}

链式栈

使用链表来实现栈,push进的数据添加至链表的头结点,pop数据时取的也是链表的头结点,这样就实现了栈的后进先出。代码如下:

public class StackBasedLinkedList implements StackInterface {
    private Node first;

    /**
     * 每次添加数据都向链表头结点添加。
     *
     * @param value
     * @return
     */
    @Override
    public Boolean push(String value) {
        Node newNode = new Node(value, null);
        if (null == first) {
            first = newNode;
        } else {
            newNode.next = first;
            first = newNode;
        }
        return true;
    }

    /**
     * 每次获取数据都从链表头结点获取。
     *
     * @return
     */
    @Override
    public String pop() {
        if (null == first) {
            return null;
        } else {
            String tmp = first.getValue();
            first = first.next;
            return tmp;
        }
    }

    @Override
    public String toString() {
        StringBuilder sb = new StringBuilder();
        sb.append('[');
        if (first != null) {
            Node curr = first;
            while (curr != null) {
                sb.append(String.valueOf(curr));
                if (curr.next != null) {
                    sb.append(",").append(" ");
                }
                curr = curr.next;
            }
        }
        sb.append("]");
        return sb.toString();
    }

    private static class Node {
        Node next;
        String value;

        public Node(String value, Node next) {
            this.next = next;
            this.value = value;
        }

        public String getValue() {
            return value;
        }

        @Override
        public String toString() {
            return value;
        }
    }
}

测试代码:

    StackBasedLinkedList sbll = new StackBasedLinkedList();
    System.out.println(sbll);

    // 添加数据
    for (int i = 0; i < 10; i++) {
        sbll.push("" + i + i + i);
    }
    System.out.println(sbll);

    // 获取数据
    for (int i = 0; i < 11; i++) {
        System.out.println("sbll.pop():" + sbll.pop());
    }

输出结果:符合预期。

[]
[999, 888, 777, 666, 555, 444, 333, 222, 111, 000]
sbll.pop():999
sbll.pop():888
sbll.pop():777
sbll.pop():666
sbll.pop():555
sbll.pop():444
sbll.pop():333
sbll.pop():222
sbll.pop():111
sbll.pop():000
sbll.pop():null

完整代码请查看

项目中搜索SingleLinkedList即可。

github传送门 https://github.com/tinyvampirepudge/DataStructureDemo

gitee传送门 https://gitee.com/tinytongtong/DataStructureDemo

参考:
如何实现浏览器的前进和后退功能?

相关文章
|
2月前
|
Java
在Java中实现接口的具体代码示例
可以根据具体的需求,创建更多的类来实现这个接口,以满足不同形状的计算需求。希望这个示例对你理解在 Java 中如何实现接口有所帮助。
92 38
|
12天前
|
安全 Java 编译器
深入理解Java中synchronized三种使用方式:助您写出线程安全的代码
`synchronized` 是 Java 中的关键字,用于实现线程同步,确保多个线程互斥访问共享资源。它通过内置的监视器锁机制,防止多个线程同时执行被 `synchronized` 修饰的方法或代码块。`synchronized` 可以修饰非静态方法、静态方法和代码块,分别锁定实例对象、类对象或指定的对象。其底层原理基于 JVM 的指令和对象的监视器,JDK 1.6 后引入了偏向锁、轻量级锁等优化措施,提高了性能。
35 3
|
2月前
|
Java
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
56 24
|
20天前
|
前端开发 Java 测试技术
java日常开发中如何写出优雅的好维护的代码
代码可读性太差,实际是给团队后续开发中埋坑,优化在平时,没有那个团队会说我专门给你一个月来优化之前的代码,所以在日常开发中就要多注意可读性问题,不要写出几天之后自己都看不懂的代码。
56 2
|
1月前
|
Java 编译器 数据库
Java 中的注解(Annotations):代码中的 “元数据” 魔法
Java注解是代码中的“元数据”标签,不直接参与业务逻辑,但在编译或运行时提供重要信息。本文介绍了注解的基础语法、内置注解的应用场景,以及如何自定义注解和结合AOP技术实现方法执行日志记录,展示了注解在提升代码质量、简化开发流程和增强程序功能方面的强大作用。
81 5
|
1月前
|
存储 算法 Java
Java 内存管理与优化:掌控堆与栈,雕琢高效代码
Java内存管理与优化是提升程序性能的关键。掌握堆与栈的运作机制,学习如何有效管理内存资源,雕琢出更加高效的代码,是每个Java开发者必备的技能。
57 5
|
2月前
|
Java API 开发者
Java中的Lambda表达式:简洁代码的利器####
本文探讨了Java中Lambda表达式的概念、用途及其在简化代码和提高开发效率方面的显著作用。通过具体实例,展示了Lambda表达式如何在Java 8及更高版本中替代传统的匿名内部类,使代码更加简洁易读。文章还简要介绍了Lambda表达式的语法和常见用法,帮助开发者更好地理解和应用这一强大的工具。 ####
|
2月前
|
Java API Maven
商汤人像如何对接?Java代码如何写?
商汤人像如何对接?Java代码如何写?
50 5
|
1月前
|
安全 Java API
Java中的Lambda表达式:简化代码的现代魔法
在Java 8的发布中,Lambda表达式的引入无疑是一场编程范式的革命。它不仅让代码变得更加简洁,还使得函数式编程在Java中成为可能。本文将深入探讨Lambda表达式如何改变我们编写和维护Java代码的方式,以及它是如何提升我们编码效率的。
|
2月前
|
Java
Java将OffsetDateTime格式化为 yyyy-MM-dd HH:mm:ss 如何写代码?
Java将OffsetDateTime格式化为 yyyy-MM-dd HH:mm:ss 如何写代码?
38 0