java数据结构基础:栈

目录
  • 准备工作
  • 编码环节
    • push方法
    • pop方法
    • empty方法
    • 全部代码
  • 总结

准备工作

工具:idea+jdk8

技术要求:java基础语法

编码环节

首先,我们得先确定下来,用什么数据来模拟栈的操作。由于是一个一个的元素放入栈里面,我们可以考虑用数组来实现。

以上是Java官方文档中的栈定义,我们也只需要实现三个方法:判断是否为空、移除栈顶对象、添加元素到栈的尾部

所以我们事先得定义一个数组:

Objects[] arr;

数组定义好了之后呢,想想,我们怎么去获取到栈尾部或者栈首的元素呢?还记得数组的索引吗?可以用索引来假设为栈的指针。所以,我们还得定义好栈的元素个数和栈的默认长度以及默认的指针:

private int stackLength = 4; // 数组的默认长度
private int size; // 记住栈容器的元素个数
private int index = -1; // 操作数组下标位置的指针

为什么这儿指向的是-1呢?我们知道,数组的第一个元素是索引为0,那么-1的意思就是不指向任何元素。待会儿我们在用的时候再去指向他。

然后,我们还得定义出数组的初始化。以及初始化的长度。参考官方文档的写法,当栈的长度满了之后我们就对栈长度进行1.5倍的扩容。我们就单独提取出一个方法来放置;

/**
 * 数组初始化或者以1.5倍容量对数组扩容
 */
private void capacity() {
    // 数组初始化
    if (this.arr == null) {
        this.arr = new Object[this.stackLength];
    }
    // 以1.5倍对数组扩容
    if (this.size - (this.stackLength - 1) >= 0) { // 如果当前数组的元素个数大于了当前数组的最后一个索引值
        this.stackLength = this.stackLength + (this.stackLength >> 1); // 位运算,让长度变成原来的1/2
        this.arr = Arrays.copyOf(this.arr, this.stackLength); // 复制一个新的数组,用新开辟的长度
    }
}

push方法

如何给栈添加元素?我们要考虑的地方:指针向右移动一位,也就是说指针要+1。其次,添加完元素之后,栈元素的长度发生了变化,size+1 。

public E push(E item){
    // 先初始化数组
    this.capacity();
    // 添加元素
    this.arr[++index] = item;
    // 记录元素个数加一
    this.size++;
    return item;
}

pop方法

pop方法主要是用来移除栈顶的元素。

先分析一下思路:我们要用index去指向栈顶的元素,该怎么去指定?

删除之后,对应的size长度该怎么去改变?

我们知道,当元素添加了之后,index会跟着改变,那么就好比我们添加了三个元素,此时的index应该就是指向的2。那就好办了。

当移除的时候,我们只需要让index–来操作就能解决问题;看代码:

/**
 * 获取栈顶元素
 *
 * @return
 */
public E pop() {
    // 如果栈容器中没有元素则抛出异常
    if (this.index == -1) {
        throw new EmptyStackException();
    }
    // 记录元素个数
    this.size--;
    // 返回栈顶元素
    System.out.println("删除元素之前的当前下标:"+index);
    return (E) this.arr[index--];
}

empty方法

判断栈是否为空,这很简单。直接判断当前的size是不是0就能解决:

public boolean empty(){
	return this.index==0?true:false;
}

全部代码

package com.zxy;
import java.util.Arrays;
import java.util.EmptyStackException;
/**
 * @Author Zxy
 * @Date 2021/2/2 20:24
 * @Version 1.0
 * 演示栈容器的使用
 */
public class MyStack<E> {
    private Object[] arr; // 存放元素的物理结构
    private int stackLength = 4; // 数组的默认长度
    private int size; // 记住栈容器的元素个数
    private int index = -1; // 操作数组下标位置的指针
    /**
     * 判断栈容器是否为空
     */
    public boolean empty() {
        return this.size == 0 ? true : false;
    }
    /**
     * 获取栈顶元素
     *
     * @return
     */
    public E pop() {
        // 如果栈容器中没有元素则抛出异常
        if (this.index == -1) {
            throw new EmptyStackException();
        }
        // 记录元素个数
        this.size--;
        // 返回栈顶元素
        System.out.println("删除元素之前的当前下标:"+index);
        return (E) this.arr[index--];
    }

    /**
     * 向栈顶添加元素
     *
     * @param item
     * @return
     */
    public E push(E item) {
        // 初始化数组
        this.capacity();
        // 向数组中添加元素
        System.out.println("添加元素之前的下标:"+index);
        this.arr[++index] = item;
        System.out.println("添加元素之后的下标:"+index);
        // 记录元素个数
        this.size++;
        return item;
    }
    /**
     * 数组初始化或者以1.5倍容量对数组扩容
     */
    private void capacity() {
        // 数组初始化
        if (this.arr == null) {
            this.arr = new Object[this.stackLength];
        }
        // 以1.5倍对数组扩容
        if (this.size - (this.stackLength - 1) >= 0) { // 如果当前数组的元素个数大于了当前数组的最后一个索引值
            this.stackLength = this.stackLength + (this.stackLength >> 1); // 位运算,让长度变成原来的1/2
            this.arr = Arrays.copyOf(this.arr, this.stackLength); // 复制一个新的数组,用新开辟的长度
        }
    }
    public static void main(String[] args) {
        MyStack<String> stack = new MyStack<>();
        stack.push("a");
        stack.push("b");
        stack.push("c");
        System.out.println(stack.size);
        System.out.println("当前栈顶元素:"+stack.pop());
        /*System.out.println(stack.pop());
        System.out.println(stack.pop());*/
    }
}

总结

本篇文章就到这里了,希望能给你带来帮助,也希望能够您能够关注我们的更多内容!

(0)

相关推荐

  • java数据结构基础:单,双向链表

    目录 单向链表 单链表图解 代码 双向链表 编码 总结 单向链表 单向链表比顺序结构的线性表最大的好处就是不用保证存放的位置,它只需要用指针去指向下一个元素就能搞定. 单链表图解 图画的比较粗糙,简单的讲解一下: 上面四个长方形,每个长方形都是一个节点.在长方形中,一种包含两个东西,一个是当前节点的元素,一个是指向下一节点的地址.这个下一个节点的地址指向了下一个节点中的元素.以此类推. 在最左边的叫做头节点,同样,最后面的叫尾节点. 所以,我们所有的操作都是根据节点来进行操作. 代码 这些代码都

  • java数据结构基础:算法

    目录 数据结构和算法关系 高斯求和 算法定义 算法的特性 算法设计的要求 算法效率的度量方法 函数的渐进增长 总结 数据结构和算法关系 虽然这个标题起的叫数据结构,但是我却总结算法...我不是没事找抽,只是呢,在学数据结构的时候,算法是你肯定离不开的东西. 你平时在网上看到的那些文章,在你不经意间搜的时候,是不是都是搜的数据结构与算法这七个字.这说明啥,这说明他们俩是离不开的. 给你打个比方,你想看德云社相声(我也想看),有一天你最想看小岳岳专场,想看小白专场.但是呢,走到园子里之后发现,他们今

  • Java常见基础数据结构

    目录 栈: 队列: 数组: 链表: 红黑树: 总结 栈: stack,又称堆栈,他是运算受限的线性表,其限制是仅允许在表的一端进行插入和删除操作,不允许在其他任何位置进行添加.查找.删除等操作. 简单的来说,采用该结构的集合,对元素的存取有如下几个特点 1.先进后出. 2.栈的入口.出口都是栈的顶端位置. 压栈:就是存元素,把元素存储到栈的顶端位置,栈中已有元素一次向栈底方向移动一个位置. 弹栈:就是取元素,把栈顶端的元素取出,栈中已有元素依次向栈顶方向移动一个位置. 队列: queue,简称队

  • java数据结构基础:绪论

    目录 基本概念和术语 数据 数据元素 数据项 数据对象 结构 数据结构 逻辑结构与物理结构 逻辑结构 物理结构 抽象数据类型 总结 基本概念和术语 要想知道数据结构是什么,我们首先得去知道,数据和结构是什么: 数据结构=数据+结构 也就是说,我们先去研究数据,再去把这些数据组成一定得样子(结构),自然而然的成了数据结构 数据 数据:是描述客观事物的符号,是计算机中可以操作的对象,是能被计算机识别并输入给计算机处理的符号集合 这样说可能还是有人觉得头痛,说直白点,空气粒子组成了空气,一个个的人组成

  • java数据结构基础:线性表

    目录 前言 需求分析 编码 add方法 getIndex方法 pop方法 insert方法 getAll 全部代码 总结 前言 其实线性表在生活中和栈的结构差不多.昨天总结了一篇单链表,也是线性表的一种. 今天用另一种写法来控制指针的移动实现数据的顺序存储结构. 需求分析 首先要明确,这种顺序存储结构的线性表底层用什么.根据之前查看过的源码来看,list一般都是以数组为底层.我们也不例外. 其次,我们还得去定义好线性表的长度,以及每个元素的指针. private Object[] arr; //

  • java数据结构基础:栈

    目录 准备工作 编码环节 push方法 pop方法 empty方法 全部代码 总结 准备工作 工具:idea+jdk8 技术要求:java基础语法 编码环节 首先,我们得先确定下来,用什么数据来模拟栈的操作.由于是一个一个的元素放入栈里面,我们可以考虑用数组来实现. 以上是Java官方文档中的栈定义,我们也只需要实现三个方法:判断是否为空.移除栈顶对象.添加元素到栈的尾部 所以我们事先得定义一个数组: Objects[] arr; 数组定义好了之后呢,想想,我们怎么去获取到栈尾部或者栈首的元素呢

  • java 数据结构之栈与队列

    java 数据结构之栈与队列 一:对列 队列是一种先进先出的数据结构 实现代码: package Queue; /* * 使用java构建队列,并模拟实现队列的入队和出对方法 */ public class Queue { //队列类 private int maxSize; //定义队列的长度 private int[] arrQueue; //队列 private int rear; //定义队列的尾指针 private int front; //定义队列的头指针 private int e

  • Java数据结构之栈的基本定义与实现方法示例

    本文实例讲述了Java数据结构之栈的基本定义与实现方法.分享给大家供大家参考,具体如下: 一.概述: 1.基本概念: 栈是一种数据结构,是只能在某一端插入和删除的特殊线性表.它按照后进先出的原则存储数据,先进入的数据被压入栈底,最后的数据在栈顶,需要读数据的时候从栈顶开始弹出数据(最后一个数据被第一个读出来). 栈是允许在同一端进行插入和删除操作的特殊线性表.允许进行插入和删除操作的一端称为栈顶(top),另一端为栈底(bottom):栈底固定,而栈顶 浮动:栈中元素个数为零时称为空栈.插入一般

  • Java数据结构之栈的线性结构详解

    目录 一:栈 二:栈的实现 三:栈的测试 四:栈的应用(回文序列的判断) 总结 一:栈 栈是限制插入和删除只能在一个位置上进行的表,此位置就是表的末端,叫作栈顶. 栈的基本操作分为push(入栈) 和 pop(出栈),前者相当于插入元素到表的末端(栈顶),后者相当于删除栈顶的元素. 二:栈的实现 public class LinearStack { /** * 栈的初始默认大小为10 */ private int size = 5; /** * 指向栈顶的数组下标 */ int top = -1

  • Java数据结构之栈与队列实例详解

    目录 一,栈 1,概念 2,栈的操作 3,栈的实现  4,实现mystack 二,队列 1,概念  2,队列的实现  3,实现myqueue 栈.队列与数组的区别? 总结 一,栈 1,概念 在我们软件应用 ,栈这种后进先出数据结构的应用是非常普遍的.比如你用浏 览器上网时不管什么浏览器都有 个"后退"键,你点击后可以接访问顺序的逆序加载浏览过的网页.   很多类似的软件,比如 Word Photoshop 等文档或图像编 软件中 都有撤销 )的操作,也是用栈这种方式来实现的,当然不同的

  • java数据结构关于栈的实例应用

    此文章介绍关于顺序栈,链式栈的实例操作,括号匹配,表达式求值(后缀表达式) 1.声明一个栈接口SStack package ch05; public interface SStack <T>{ boolean isEmpty(); // 判断栈是否为空 void push(T x); // 元素x入栈 T pop(); // 出栈,返回栈顶元素 T peek(); // 返回栈顶元素,但不出栈 }  2. 定义顺序栈类SeqStack<T>,包括数据元素的对象数组和栈顶元素下标两个

  • java数据结构基础:循环链表和栈

    目录 循环链表: 实现思路: 代码实现: 栈: 实现思路: 代码实现: 总结 循环链表: 与单链表的最后一个节点的指针域为null不同,循环链表的最后一个节点的指针指向头结点 实现思路: 初始化时将头结点指向自身,添加节点到链表末尾时,将新节点的指针指向头结点 在遍历链表时,判断是否遍历到链表末尾,需要判断当前指针的下一个节点是否为头结点 代码实现: 节点类CircleNode: public class CircleNode { public int data; public CircleNo

随机推荐