我们从一个DEMO作为入口,了解Java的Stack的源码,代码如:

 Stack<String> stack = new Stack<>();
 stack.push("a"); // 入栈
 stack.peek(); // 查询栈顶元素
 stack.pop(); // 出栈
 stack.empty(); // 判空
 stack.search("a"); // 搜索元素

Stack类继承自Vector,基于数组实现,如图:

主要包含5个方法:入栈、出栈、查询栈顶、判空、查找元素,每个方法都有sychronized同步锁来保持同步,所以是线程安全的我们自上而下打开每个方法

入栈push()

入栈操作会先判断是否有足够的空间来存储新元素,如果没有的话那么要进行扩容

 public E push(E item) {
     addElement(item); // 添加元素
     return item; // 返回当前元素
 }

 public synchronized void addElement(E obj) {
     modCount++; // 修改次数+1
     ensureCapacityHelper(elementCount + 1); // 确定数组容量是否足够,如果不够那么扩充数组
     elementData[elementCount++] = obj; // 栈顶索引+1,将当前元素赋值到栈顶
 }

 private void ensureCapacityHelper(int minCapacity) {
     if (minCapacity - elementData.length > 0) // 当前所需的容量 - 当前数组的容量 如果大于0,那么意味着超出了数组的容量
         grow(minCapacity); // 超出则扩充数组容量
 }

 private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8; // 数组元素很大的时候,可能会超出预想导致内存溢出,需要限定数组的最大值;这里-8的目的是因为有的虚拟机在实现的时候会在数组头部预留一些空间

 private void grow(int minCapacity) {
     int oldCapacity = elementData.length; // 当前数组的容量
     int newCapacity = oldCapacity + ((capacityIncrement > 0) ? capacityIncrement : oldCapacity); // 计算新的容量,如果有指定增长那么 + 指定增长,如果没有指定那么当前容量乘以2
     if (newCapacity - minCapacity < 0) // 如果新容量还是不够
         newCapacity = minCapacity; // 使用所需容量为新容量
     if (newCapacity - MAX_ARRAY_SIZE > 0) // 如果新容量超过容量上限
         newCapacity = hugeCapacity(minCapacity);
     elementData = Arrays.copyOf(elementData, newCapacity); // 将原数组的数据拷贝到新的容器里,并替换原数组
 }

 private static int hugeCapacity(int minCapacity) {
     if (minCapacity < 0) // 超出INT最大值则会变为负数,这里抛出内存溢出异常
         throw new OutOfMemoryError();
     return (minCapacity > MAX_ARRAY_SIZE) ? Integer.MAX_VALUE : MAX_ARRAY_SIZE; // 所需容量大于上限,那么采用INT最大值,否则采用上限值
 } 

查询栈顶peek()

根据栈顶索引获取栈顶元素,但如果没有一个元素的情况下会抛出空栈异常

 public synchronized E peek() {
     int len = size(); // 当前栈的元素大小
     if (len == 0)  // 空栈抛出异常
         throw new EmptyStackException();
     return elementAt(len - 1); // 获取栈顶元素
 }

 public synchronized E elementAt(int index) {
     if (index >= elementCount) { // 栈顶索引大于等于元素大小表示越界
         throw new ArrayIndexOutOfBoundsException(index + " >= " + elementCount);
     }
     return elementData(index);
 }

 E elementData(int index) {
     return (E) elementData[index]; // 根据数组索引位置获取元素
 }

出栈pop()

出栈只是把栈顶的元素置为null,让GC去回收。出栈之前会先获取栈顶元素,所有如果对空栈出栈,那么会抛出异常

 public synchronized E pop() {
     E       obj;
     int     len = size(); // 获取当前元素大小
     obj = peek(); // 获取栈顶元素
     removeElementAt(len - 1); // 移除栈顶元素
     return obj; // 返回被移除的元素
 }

 public synchronized void removeElementAt(int index) {
     modCount++; // 统计修改次数
     if (index >= elementCount) { // 越界校验
         throw new ArrayIndexOutOfBoundsException(index + " >= " + elementCount);
     } else if (index < 0) { // 越界校验
         throw new ArrayIndexOutOfBoundsException(index);
     }
     int j = elementCount - index - 1;
     if (j > 0) {
         System.arraycopy(elementData, index + 1, elementData, index, j); // 这里调用native方法,把被移除的元素之后的元素向前移动一位
     }
     elementCount--; // 元素数量 -1
     elementData[elementCount] = null; // 置为null,让GC去销毁该元素
 }

判空empty()

 public boolean empty() {
     return size() == 0; // 判断元素数量为0
 }

查找元素search()

查找元素的时候从栈顶向下查找,如果有多个那么只匹配最上面的一个,没有则返回-1

 public synchronized int search(Object o) {
     int i = lastIndexOf(o); // 从栈顶向下搜索,如果匹配到则返回自上而下的索引,否则返回-1
     if (i >= 0) {
         return size() - i; // 计算自下而上的索引
     }
     return -1; // 没有找到
 }

 public synchronized int lastIndexOf(Object o) {
     return lastIndexOf(o, elementCount-1); // 索引从栈顶开始
 }

 public synchronized int lastIndexOf(Object o, int index) {
     if (index >= elementCount) // 越界检验
         throw new IndexOutOfBoundsException(index + " >= "+ elementCount);
     if (o == null) { // 如果元素为null
         for (int i = index; i >= 0; i--) // 自栈顶向下遍历
             if (elementData[i]==null) // 如果存在 == null的,返回
                 return i;
     } else { // 元素不为null
         for (int i = index; i >= 0; i--) // 自栈顶向下遍历
             if (o.equals(elementData[i])) // 元素相等的,返回
                 return i;
     }
     return -1; // 没有找到相等的元素
 }

Stack源码解析的更多相关文章

  1. Java - Stack源码解析

    Java提高篇(三一)-----Stack 在Java中Stack类表示后进先出(LIFO)的对象堆栈.栈是一种非常常见的数据结构,它采用典型的先进后出的操作方式完成的.每一个栈都包含一个栈顶,每次出 ...

  2. Java 集合系列07之 Stack详细介绍(源码解析)和使用示例

    概要 学完Vector了之后,接下来我们开始学习Stack.Stack很简单,它继承于Vector.学习方式还是和之前一样,先对Stack有个整体认识,然后再学习它的源码:最后再通过实例来学会使用它. ...

  3. Java 集合系列 06 Stack详细介绍(源码解析)和使用示例

    java 集合系列目录: Java 集合系列 01 总体框架 Java 集合系列 02 Collection架构 Java 集合系列 03 ArrayList详细介绍(源码解析)和使用示例 Java ...

  4. Java 集合系列Stack详细介绍(源码解析)和使用示例

    Stack简介 Stack是栈.它的特性是:先进后出(FILO, First In Last Out). java工具包中的Stack是继承于Vector(矢量队列)的,由于Vector是通过数组实现 ...

  5. 【转】 Java 集合系列07之 Stack详细介绍(源码解析)和使用示例

    概要 学完Vector了之后,接下来我们开始学习Stack.Stack很简单,它继承于Vector.学习方式还是和之前一样,先对Stack有个整体认识,然后再学习它的源码:最后再通过实例来学会使用它. ...

  6. Java Stack源码分析

    Stack简介 Stack是栈.它的特性是:先进后出(FILO, First In Last Out).java工具包中的Stack是继承于Vector(矢量队列)的,由于Vector是通过数组实现的 ...

  7. 给jdk写注释系列之jdk1.6容器(10)-Stack&Vector源码解析

    前面我们已经接触过几种数据结构了,有数组.链表.Hash表.红黑树(二叉查询树),今天再来看另外一种数据结构:栈.      什么是栈呢,我就不找它具体的定义了,直接举个例子,栈就相当于一个很窄的木桶 ...

  8. 源码解析-Volley(转自codeKK)

    Volley 源码解析 本文为 Android 开源项目源码解析 中 Volley 部分项目地址:Volley,分析的版本:35ce778,Demo 地址:Volley Demo分析者:grumoon ...

  9. Java 集合系列05之 LinkedList详细介绍(源码解析)和使用示例

    概要  前面,我们已经学习了ArrayList,并了解了fail-fast机制.这一章我们接着学习List的实现类——LinkedList.和学习ArrayList一样,接下来呢,我们先对Linked ...

随机推荐

  1. UWP 取消GridView、ListView鼠标选中、悬停效果

    因为经常碰到ListView或者ListBox之类的选中.鼠标悬停样式和自己设置的主题颜色不搭,这时就需要改变这些样式了. 而这里我通过ListView来说明,大致思路其实就是重新定义Item的Tem ...

  2. ubuntu下wps无法使用搜狗输入法输入中文

    sudo vim /usr/bin/et sudo vim /usr/bin/wps sudo vim /usr/bin/wpp 以上三个文件,都加入如下内容后重新打开文档即可 export XMOD ...

  3. 组件基础(非父子组件传值)—Vue学习笔记

    最近几天忙着写Api去了,抽空把后面的内容下出来,然后再分享给大家web可以使用的api. 上次说了父子组件直接的传值,这次看一下非父子组件之间的传值(总线机制) 要实现非父子组件之间的传值非常重要的 ...

  4. SQLYog执行SQL脚本提示:错误代码: 1067 - Invalid default value for '数据库表'查询:解决办法

    强烈建议:完全卸载当前版本MySQL,重新安装5.6及以上版本 完全卸载方法:https://jingyan.baidu.com/article/3d69c551611290f0ce02d77b.ht ...

  5. [工具]Tomcat CVE-2017-12615 远程代码执行

    工具: K8_TocmatExp编译: VS2012  C# (.NET Framework v2.0)组织: K8搞基大队[K8team]作者: K8拉登哥哥博客: http://qqhack8.b ...

  6. [Umbraco] 项目结构

    主要包括以下几个部分 Umbraco.businesslogic(基础) Umbraco.cms (核心) Umbraco.controls(控件) Umbraco.datalayer(数据) Umb ...

  7. 前端代码质量保障之代码review

    经验丰富的程序员和一般程序员之间的最大区别,不仅体现在解决问题的能力上, 还体现在日常代码的风格上.掌握一门技术可能需要几月,甚至几周就够了. 好的习惯风格养成却需数年. 团队成员之间需要合作,代码需 ...

  8. 解决 在 WINDOWS 下 同时安装 python2 python3 后 pip 错误

    再之前同时安装 python 后 只需把环境变量PATH 里面改为 PATH=C:\Python36-32\Scripts\;C:\Python36-32\;C:\Python27\;C:\Pytho ...

  9. Linux后台进程管理以及ctrl+z(挂起)、ctrl+c(中断)、ctrl+\(退出)和ctrl+d(EOF)的区别(转)

    一.后台进程管理命令 fg.bg.jobs.&.ctrl + z.ctrl + c.ctrl + \.ctrl + d1. &加在一个命令的最后,可以把这个命令放到后台执行 ,如fir ...

  10. Selenium自动化测试Python五:WebDriver设计模式

    WebDriver 设计模式 欢迎阅读WebDriver进阶讲义.本篇讲义将会重点介绍Selenium WebDriver 自动化框架的设计,着重使用Page Object设计模式,以及使用HTML测 ...