Java 为什么不推荐使用 Stack 呢?

  因为 Stack 是 JDK 1.0 的产物。它继承自 Vector,Vector 都不被推荐使用了,你说 Stack 还会被推荐吗?

  当初 JDK1.0 在开发时,可能为了快速的推出一些基本的数据结构操作,所以推出了一些比较粗糙的类。比如,Vector、Stack、Hashtable等。这些类中的一些方法加上了 synchronized 关键字,容易给一些初级程序员在使用上造成一些误解!而且在之前的几个版本中,性能还不怎么好。

  基于 Vector 实现的栈 Stack。底层实际上还是数组,所以还是存在需要扩容。Vector 是由数组实现的集合类,它包含了大量集合处理的方法。而 Stack 之所以继承 Vector,是为了复用 Vector 中的方法,来实现进栈(push)、出栈(pop)等操作。这里就是 Stack 设计不好的地方,既然只是为了实现栈,不用链表来单独实现,而是为了复用简单的方法而迫使它继承 Vector,Stack 和 Vector 本来是毫无关系的。这使得 Stack 在基于数组实现上效率受影响,另外因为继承 Vector 类,Stack 可以复用 Vector 大量方法,这使得 Stack 在设计上不严谨。

  Java 提供了 Deuqe。Deque 是继承自 Queue,而 Stack 是继承自 Vector。Java 中的 Deuqe,即“double ended queue”的缩写,是 Java 中的双端队列集合类型。Deque 具备普通队列 FIFO 的功能,同时它也具备了 Stack 的 LIFO 功能,并且保留了 push 和 pop 函数,所以使用起来应该是一点障碍都没有。

  ArrayDeque 是 Deque 接口的一种具体实现,是依赖于可变数组来实现的。ArrayDeque 没有容量限制,可根据需求自动进行扩容。ArrayDeque 可以作为栈来使用,效率要高于 Stack。ArrayDeque 也可以作为队列来使用,效率相较于基于双向链表的 LinkedList 也要更好一些。注意,ArrayDeque 不支持为 null 的元素。

原文链接:https://www.xttblog.com/?p=3416

关于java.util.stack的底层实现

当push一个元素时:

public E push(E item) {
addElement(item);
return item;
}

实际上调用但是Vector的addElement方法

public synchronized void addElement(E obj) {
modCount++;
ensureCapacityHelper(elementCount + 1);
elementData[elementCount++] = obj;
}

而Vector的底层实现就是一个数组,初始大小为10,每当元素超出就增长为原来的2倍,和ArrayList的增长方式类似。

public class Vector<E>
extends AbstractList<E>
implements List<E>, RandomAccess, Cloneable, java.io.Serializable{
protected Object[] elementData;
public Vector() {
//初始大小为10
this(10);
}
public Vector(int initialCapacity) {
this(initialCapacity, 0);
}
public Vector(int initialCapacity, int capacityIncrement) {
super();
if (initialCapacity < 0)
throw new IllegalArgumentException("Illegal Capacity: "+
initialCapacity);
//初始化为10
this.elementData = new Object[initialCapacity];
this.capacityIncrement = capacityIncrement;
……
}

  总结:在java.util.stack中,栈的底层是使用数组来实现的,数组初始大小为10。每当元素个数超出数组容量就扩展为原来的2倍,将原数组中的元素拷贝到新数组中,随着数组元素的增多,这种开销也越大。

​ Java并不推荐使用java.util.stack来进行栈的操作,而是推荐使用一个双端队列“deque ”来代替它。

关于“deque”

public interface Deque<E>extends Queue<E>

  它是一个线性 collection,支持在两端插入和移除元素。名称 deque 是“double ended queue(双端队列)”的缩写,通常读为“deck”。大多数 Deque 实现对于它们能够包含的元素数没有固定限制,但此接口既支持有容量限制的双端队列,也支持没有固定大小限制的双端队列。

  此接口定义在双端队列两端访问元素的方法。提供插入、移除和检查元素的方法。每种方法都存在两种形式:一种形式在操作失败时抛出异常,另一种形式返回一个特殊值(null 或 false,具体取决于操作)。插入操作的后一种形式是专为使用有容量限制的 Deque 实现设计的;在大多数实现中,插入操作不能失败。

  双端队列也可用作 LIFO(后进先出)堆栈。应优先使用此接口而不是遗留 Stack 类。在将双端队列用作堆栈时,元素被推入双端队列的开头并从双端队列开头弹出。堆栈方法完全等效于 Deque 方法

最新文章

  1. datatables中的Options总结(3)
  2. android wifi P2P CONNECT, INVITE和JOIN流程选择
  3. domReady方法(dom加载完成执行回调)
  4. Ubuntu上部署一个简单的Java项目
  5. html5开发之viewport使用
  6. virtualenv 和 virtualenvwrapper 实践
  7. MVC-登录并设置角色
  8. [网络]让局域网的电脑通过路由的公网ip可以被访问到的设置
  9. 11.3 morning
  10. LeetCode Javascript实现 283. Move Zeroes 349. Intersection of Two Arrays 237. Delete Node in a Linked List
  11. [20190415]10g下那些latch是共享的.txt
  12. 【转载】 第四范式涂威威:AutoML技术现状与未来展望
  13. Codeforces 438E The Child and Binary Tree [DP,生成函数,NTT]
  14. WxWidgets笔记
  15. Android ThreadPoolExecutor线程池
  16. Dubbo高级篇4
  17. 【Python】创建xml文档
  18. java接口的学习笔记
  19. 解决mysql安装出现error Nr.1045问题
  20. java中二维数组遍历

热门文章

  1. Cypress系列(41)- Cypress 的测试报告
  2. 基于Bilateral Attention和Pyramid Filling Block的图像修复方法
  3. onepill服务端
  4. Redis之NoSql入门和概述(一)
  5. 递归-N皇后问题
  6. PHP xml_set_element_handler() 函数
  7. PDOStatement::debugDumpParams
  8. 牛客挑战赛40 VMware和基站 set 二分 启发式合并 区间覆盖
  9. ZROI 提高十连测 DAY3
  10. 区块链钱包开发 - USDT - 一、Omni本地钱包安装