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

简介: 杂谈"栈"结构:     栈(Stack)是一种插入删除操作都只能在一个位置上进表,这个位置位于表的末端,叫做栈顶(Top).   对栈的基本操作有push和pop,表示进栈和出栈.也就相当于插入和删除操作.

杂谈"栈"结构:  

  栈(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取余操作,然后将余数压栈,之后再将所有的数出栈,即是所对应的二进制数,这其实是栈对于逆序操作的一个实例.

  平衡符号:

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

    实现过程大概如下:

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

  后缀表达式:

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

  用于方法调用:

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

 

相关文章
|
6月前
|
存储 Java 编译器
深入理解Java虚拟机--类文件结构
本内容介绍了Java虚拟机与Class文件的关系及其内部结构。Class文件是一种与语言无关的二进制格式,包含JVM指令集、符号表等信息。无论使用何种语言,只要能生成符合规范的Class文件,即可在JVM上运行。文章详细解析了Class文件的组成,包括魔数、版本号、常量池、访问标志、类索引、字段表、方法表和属性表等,并说明其在Java编译与运行过程中的作用。
173 0
|
10月前
|
前端开发 Cloud Native Java
Java||Springboot读取本地目录的文件和文件结构,读取服务器文档目录数据供前端渲染的API实现
博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
Java||Springboot读取本地目录的文件和文件结构,读取服务器文档目录数据供前端渲染的API实现
|
12月前
|
存储 Java 开发者
【潜意识Java】深入详细理解分析Java中的toString()方法重写完整笔记总结,超级详细。
本文详细介绍了 Java 中 `toString()` 方法的重写技巧及其重要
703 10
【潜意识Java】深入详细理解分析Java中的toString()方法重写完整笔记总结,超级详细。
|
10月前
|
传感器 监控 Java
Java代码结构解析:类、方法、主函数(1分钟解剖室)
### Java代码结构简介 掌握Java代码结构如同拥有程序世界的建筑蓝图,类、方法和主函数构成“黄金三角”。类是独立的容器,承载成员变量和方法;方法实现特定功能,参数控制输入环境;主函数是程序入口。常见错误包括类名与文件名不匹配、忘记static修饰符和花括号未闭合。通过实战案例学习电商系统、游戏角色控制和物联网设备监控,理解类的作用、方法类型和主函数任务,避免典型错误,逐步提升编程能力。 **脑图速记法**:类如太空站,方法即舱段;main是发射台,static不能换;文件名对仗,括号要成双;参数是坐标,void不返航。
422 5
|
11月前
|
存储 人工智能 算法
C 408—《数据结构》算法题基础篇—数组(通俗易懂)
408考研——《数据结构》算法题基础篇之数组。(408算法题的入门)
720 23
|
11月前
|
存储 IDE Java
java设置栈内存大小
在Java应用中合理设置栈内存大小是确保程序稳定性和性能的重要措施。通过JVM参数 `-Xss`,可以灵活调整栈内存大小,以适应不同的应用场景。本文介绍了设置栈内存大小的方法、应用场景和注意事项,希望能帮助开发者更好地管理Java应用的内存资源。
601 4
|
10月前
|
人工智能 JSON Java
列表结构与树结构转换分析与工具类封装(java版)
本文介绍了将线性列表转换为树形结构的实现方法及工具类封装。核心思路是先获取所有根节点,将其余节点作为子节点,通过递归构建每个根节点的子节点。关键在于节点需包含 `id`、`parentId` 和 `children` 三个属性。文中提供了两种封装方式:一是基于基类 `BaseTree` 的通用工具类,二是使用函数式接口实现更灵活的方式。推荐使用后者,因其避免了继承限制,更具扩展性。代码示例中使用了 Jackson 库进行 JSON 格式化输出,便于结果展示。最后总结指出,理解原理是进一步优化和封装的基础。
335 0
|
12月前
|
前端开发 JavaScript Java
Java构建工具-maven的复习笔记【适用于复习】
这篇文档由「潜意识Java」创作,主要介绍Maven的相关知识。内容涵盖Maven的基本概念、作用、项目导入步骤、依赖管理(包括依赖配置、代码示例、总结)、依赖传递、依赖范围以及依赖的生命周期等七个方面。作者擅长前端开发,秉持“得之坦然,失之淡然”的座右铭。期待您的点赞、关注和收藏,这将是作者持续创作的动力! [个人主页](https://blog.csdn.net/weixin_73355603?spm=1000.2115.3001.5343)
390 3
|
安全 Java 编译器
Kotlin教程笔记(27) -Kotlin 与 Java 共存(二)
Kotlin教程笔记(27) -Kotlin 与 Java 共存(二)