顺序栈与链式栈的图解与实现

# 顺序栈与链式栈的图解与实现java

  • 栈是一种特殊的线性表,它与线性表的区别体如今增删操做上
  • 栈的特色是先进后出,后进先出,也就是说栈的数据操做只能发生在末端,而不容许在中间节点进行操做

  • 如上图所示,对栈的增删操做都只能在末端也就是栈顶操做,
  • 栈既然是线性表那么就存在表头和表尾,不过在栈结构中,对其都进行限制改造,表尾用来输入数据也叫作栈顶(top),相应的 表头就是栈底(bottom),栈顶和栈顶是两个指针用来表示这个栈
  • 与线性表相似,栈也是又顺序表示和链式表示,分别称做顺序栈和链栈

栈的基本操做

  • 如何经过栈这个后进先出的线性表,来实现增删查呢?
  • 初始时,栈内没有数据,即空栈。此时栈顶就是栈底。
  • 当存入数据时,最早放入的数据会进入栈底。接着加入的数据都会放入到栈顶的位置。
  • 若是要删除数据,也只能经过访问栈顶的数据并删除。对于栈的新增操做,一般也叫做 push 或压栈。
  • 对于栈的删除操做,一般也叫做 pop或出栈。对于压栈和出栈,咱们分别基于顺序栈和链栈来分析

顺序栈

  • 顺序栈即就是顺序存储元素的,一般顺序栈咱们能够经过数组来实现,将数组的首元素放在栈底,最后一个元素放在栈顶,以后指定一个 top 指针指向栈顶元素的位置
  • 当栈中只有一个元素是,此时 top=0 ,通常以 top 是否为 -1 来断定是否为空栈,当定义了栈的最大容量时,则栈顶 top 必须小于最大容量值
  • 下面咱们经过 Java 代码实现一个顺序栈,很是简单以下:
/**
 * @url: i-code.online
 * @author: 云栖简码
 * @time: 2020/12/8 16:48
 */
public class Stack<T> {

    private Object[] stack;
    private int stackSize;
    private int top = -1;

    public Stack(int size){
        stackSize = size;
        stack = new Object[size];
    }

    public void push(T value){
        if (top < stackSize-1){
            top++;
            stack[top] = value;
            return;
        }
        throw new ArrayIndexOutOfBoundsException(top +"越界");
    }

    public T pop(){
        if (top > -1){
            top--;
           return (T) stack[top+1];
        }
        throw new ArrayIndexOutOfBoundsException(top +"越界");
    }

    public boolean empty(){
        return top == -1;
    }

}
  • 当须要新增数据元素,即入栈操做时,就须要将新插入元素放在栈顶,并将栈顶指针增长 1。以下图所示:
    数据结构 (1).png
  • 删除数据元素,即出栈操做,只须要 top-1 就能够了。

对于查找操做,栈没有额外的改变,跟线性表同样,它也须要遍历整个栈来完成基于某些条件的数值查找,上述代码中并未去实现该功能node

链栈

  • 关于链式栈,就是用链表的方式对栈的表示。一般,能够把栈顶放在单链表的头部,以下图所示。因为链栈的后进先出,原来的头指针就显得毫无做用了。所以,对于链栈来讲,是不须要头指针的。相反,它须要增长指向栈顶的 top 指针,这是压栈和出栈操做的重要支持。

数据结构 (2).png

  • 对于链表咱们添加都是在其后追加,可是对于链栈,新增数据的压栈操做须要额外处理的,就是栈的 top 指针。以下图所示,插入新的数据放在头部,则须要让新的结点指向原栈顶,即 top 指针指向的对象,再让 top 指针指向新的结点。

数据结构 (3).png

  • 在链式栈中进行删除操做时,只能在栈顶进行操做。所以,将栈顶的 top 指针指向栈顶元素的 next 指针便可完成删除。对于链式栈来讲,新增删除数据元素没有任何循环操做,其时间复杂度均为 O(1)
  • 经过代码简单实现栈的操做,以下:
/**
 * @url: i-code.online
 * @author: 云栖简码
 * @time: 2020/12/8 20:57
 */
public class LinkedList<E> {


    private Node<E> top = new Node<>(null,null);


    public void push(E e){
        Node<E> node = new Node<>(e,top.next);
        top.next = node;
    }

    public E pop(){
        if (top.next == null){
            throw new NoSuchElementException();
        }
        final Node<E> next = top.next;
        top.next = next.next;
        return next.item;
    }



    private static class Node<E>{
        E item;
        Node<E> next;

        public Node(E item, Node<E> next){
            this.item = item;
            this.next = next;
        }
    }

}

对于查找操做,相对链表而言,链栈没有额外的改变,它也须要遍历整个栈来完成基于某些条件的数值查找。面试

  • 无论是顺序栈仍是链栈,数据的新增、删除、查找与线性表的操做原理极为类似,时间复杂度彻底同样,都依赖当前位置的指针来进行数据对象的操做。区别仅仅在于新增和删除的对象,只能是栈顶的数据结点。

栈的案例

  • 咱们能够经过一个案例来看栈的具体使用,这里选取 leetcode 上的案例来练习,以下

有效括号

  • 给定一个只包括 '(',')','{','}','[',']' 的字符串,判断字符串是否有效。有效字符串需知足:左括号必须与相同类型的右括号匹配,左括号必须以正确的顺序匹配。例如,{ [ ( ) ( ) ] } 是合法的,而{ ( [ ) ] }是非法的。
  • 这个问题很适合采用栈来处理。缘由是,在匹配括号是否合法时,左括号是从左到右依次出现,而右括号则须要按照“后进先出”的顺序依次与左括号匹配。所以,实现方案就是经过栈的进出来完成。
  • 具体的实现思路,咱们能够遍历字符串从左起,当遇到左括号时进行压榨操做,而到遇到右括号时则继续出栈,判断出栈的括号是否与当前的右括号是一对,若是不是则非法,若是一致则继续遍历直到结束
  • 代码以下:
public boolean isValid(String s) {
        Stack stack = new Stack();
        for(int i =0;i<s.length();i++){
            char curr = s.charAt(i);
            if (isLeft(curr)) {
                stack.push(curr);
            }else {
                if (stack.empty())
                    return false;
                if (!isPair(curr,(char)stack.pop())){
                    return false;
                }
            }
        }
        if (stack.empty()){
            return true;
        }else {
            return false;
        }

    }

    public boolean isPair(char curr,char expt){
        if ((expt == '[' && curr == ']') || (expt == '{' && curr == '}') || (expt == '(' && curr == ')'))
            return true;
        return false;
    }

    public boolean isLeft(char c){
        if (c == '{' || c == '[' || c == '(')
            return true;
        return false;
    }

总结

  • 栈继承了线性表特性,是一个特殊的线性表
  • 栈只容许数据从栈顶进出,即栈的特性先进后出
  • 无论是顺序栈仍是链式栈,它们对于数据的新增操做和删除操做的时间复杂度都是 O(1)。而在查找操做中,栈和线性表同样只能经过全局遍历的方式进行,也就是须要 O(n) 的时间复杂度
  • 当咱们面临频繁增删节点,同时数据顺序有后来居上的特色时栈就是个不错的选择。例如,浏览器的前进和后退,括号匹配等问题

推荐阅读



本文由AnonyStar 发布,可转载但需声明原文出处。
欢迎关注微信公帐号 :云栖简码 获取更多优质文章
更多文章关注笔者博客 :云栖简码 i-code.online编程

相关文章
相关标签/搜索