数据结构笔记--栈的总结及java数组实现简单栈结构

杂谈"栈"结构:  

  栈(Stack)是一种插入删除操作都只能在一个位置上进表,这个位置位于表的末端,叫做栈顶(Top).

  对栈的基本操作有push和pop,表示进栈和出栈.也就相当于插入和删除操作.

  栈结构又叫做LIFO(后进先出)表.归根结底是一个表结构,因此任何能够实现表结构的方法都能实现栈.

  在java语言中,ArrayList和LinkedList都支持栈操作,栈操作都是常数时间的操作,栈的实现方式一般有两种,一种是使用顺序存储的方式,即使用数组来实现,用ArrayList可以轻易实现栈结构,也可以自己使用数组来实现,一会下面我就用数组来实现栈,第二种是使用链式存储实现,即可以使用LinkedList来实现.

使用数组实现顺序栈:

  使用arrayList和linkedList实现栈比较简单,毕竟本身他们就是封装好的功能,接下来我用数组来实现一个栈结构:

MyStack:

package com.wang.list;

import java.util.Arrays;

public class MyStack<T> {

    //使用数组实现这个栈结构
    private T[] dataArr;
    //当前元素的个数
    private int theSize;
    //栈的容量
    private static final int DEFAULT_CAPACITY=10;
    
    public MyStack(){
        clear();
        
    }
    //初始化数组,默认大小10,元素个数theSize初始化为o
    private void clear(){
        theSize=0;
        ensureCapacity(DEFAULT_CAPACITY);
    }
    
    //栈元素容量
    public int size(){
        return theSize;
    }
    
    private void ensureCapacity(int newCapacity){
        if(newCapacity<theSize){
            return;
        }
        T[] oldArr=dataArr;
        dataArr=(T[])new Object[newCapacity];
        for(int i=0;i<size();i++){
            dataArr[i]=oldArr[i];
        }
        
    }
    //入栈
    public void push(T value){
        if(dataArr.length==size()){
            ensureCapacity(size()*2);
        }
        dataArr[theSize++]=value;
    }
    //栈是否为空
    public boolean isEmpty(){
        return size()==0;
    }
    //出栈
    public T pop(){
        if(isEmpty()){
            return null;
        }
        T theValue=dataArr[theSize-1];
        dataArr[--theSize]=null;
        return theValue;
        
    }
    //返回栈尾元素
    public T peek(){
        if(isEmpty()){
            return null;
        }
        T theValue=dataArr[theSize-1];
        return theValue;
    } 
}

使用Node()辅助类实现链式栈:

 

package com.wang.list;

public class MyStack1<T> {

    
    private class Node{
        
         T data;
        Node next;
        
        
        public Node(T data,Node next){
            
            this.data=data;
            this.next=next;
        }
    }
    
    //保存元素个数
    private int theSize;
    
    //保存栈顶元素
    private Node top;
    
    public MyStack1(){
        top=null;
    }
    
    public MyStack1(T value){
        top=new Node(value,null);
    }
    
    public void push(T value){
        top=new Node(value,top);
        theSize++;
        
    }
    
    public T pop(){
        Node old=top;
        top=top.next;
        old.next=null;
        theSize--;
        return old.data;
        
    }
    
    public T peek(){
        return top.data;
        
    }
    public int size(){
        return theSize;
    }
    
    public boolean isEmpty(){
        return size()==0;
    }
}

 

 

 

栈的应用:

  进制转换:

    比如将十进制下的某个数转换为二进制中的某个数,则可以对该数进行除2取余操作,然后将余数压栈,之后再将所有的数出栈,即是所对应的二进制数,这其实是栈对于逆序操作的一个实例.

  平衡符号:

    检查那些成对出现的符号是否匹配,比如(),[],{}等

    实现过程大概如下:

      做一个空栈,读入字符直到文件末尾.如果字符是一个开放符号(比如"{}"中的"{"),则将其推入栈中.如果字符是一个封闭符号(比如"{}"中的"}"),则判断栈是否为空,为空则报错.不为空,则将栈顶元素弹出,判断弹出元素是否是其对应的开放符号,不是则报错,在文件结尾,如果栈非空,就报错.

  后缀表达式:

    不知道何为后缀表达式,请自行百度,后缀表达式的记法又称为逆波兰式,它的求值过程恰好就是从左到右的过程,可以使用一个栈,当见到一个数时就入栈,当遇到一个一个运算符时,就从栈中弹出两个数进行计算,再将所得结果入栈.最后的到的数就是计算结果.

  用于方法调用:

    存在方法调用时,比如在一个方法中调用了另一个方法,这时候需要把当前方法一些重要信息记录并保存下来,保存到一个栈中,然后再跳到新方法中去执行,当方法返回的时候,去查看栈顶的那个保存信息(栈帧),然后进行复原,事实上这是在计算机系统中一个非常重要的应用,上面的全部工作都可以用一个栈来实现.事实上,在实现递归的每一种程序语言都是这么干的,所存储的信息叫做活动记录,或者说栈帧.这个细说,比较复杂,想深入了解,自己百度吧.

 

posted @ 2016-05-11 19:50  冬至饮雪  阅读(3220)  评论(0编辑  收藏  举报