Java中栈实现怎么选?Stack、Deque、ArrayDeque、LinkedList(含常用Api积累)

news2025/1/10 16:40:43

目录

Java中的Stack类

不用Stack有以下两点原因

1、从性能上来说应该使用Deque代替Stack。

2、Stack从Vector继承是个历史遗留问题,JDK官方已建议优先使用Deque的实现类来代替Stack。

该用ArrayDeque还是LinkedList?

ArrayDeque与LinkList区别:

ArrayDeque:

LinkList:

结论

API积累

Deque中常用方法:

把Deque当栈用的时候:

把Deque当队列用的时候:

从上面(头部)插入:

从上面(头部)出来/观察:

从下面(尾部)插入:

从下面(尾部)出来/观察:

Java中的Stack类

Java中Stack类从Vector类继承,底层是用数组实现的线程安全的栈。栈是一种后进先出(LIFO)的容器,常用的操作push/pop/peek

不过Java中用来表达栈的功能(push/pop/peek),更适用的是使用双端队列接口Deque,并用实现类ArrayDequeLinkedList来进行初始化。

Deque<Integer> stack = new ArrayDeque<>();
Deque<Integer> stack = new LinkedList<>();

不用Stack有以下两点原因

1、从性能上来说应该使用Deque代替Stack。

Stack和Vector都是线程安全的,其实多数情况下并不需要做到线程安全,因此没有必要使用Stack。毕竟保证线程安全需要上锁,有额外的系统开销。

2、Stack从Vector继承是个历史遗留问题,JDK官方已建议优先使用Deque的实现类来代替Stack。

Stack从Vector继承的一个副作用是,暴露了set/get方法,可以进行随机位置的访问,这与Stack只能从尾巴上进行增减的本意相悖。

此外,Deque在转成ArrayList或者stream的时候保持了“后进先出”的语义,而Stack因为是从Vector继承,没有这个语义。

Stack<Integer> stack = new Stack<>();
Deque<Integer> deque = new ArrayDeque<>();

stack.push(1);
stack.push(2);
deque.push(1);
deque.push(2);

System.out.println(new ArrayList<>(stack)); // [1,2]
List<Integer> list1 = stack.stream().collect(Collectors.toList());//[1,2]

// deque转成ArrayList或stream时保留了“后进先出”的语义
System.out.println(new ArrayList<>(deque)); // [2,1]
List<Integer> list2 = deque.stream().collect(Collectors.toList());//[2,1]

该用ArrayDeque还是LinkedList?

ArrayDeque和LinkedList这两者底层,一个采用数组存储,一个采用链表存储;

ArrayDeque与LinkList区别:

ArrayDeque:

  • 数组结构
  • 插入元素不能为null
  • 无法确定数据量时,后期扩容会影响效率

LinkList:

  • 链表结构
  • 插入元素能为null
  • 无法确定数据量时,有更好表现

PS:这两者既可当成(仅支持在尾部加入或移除元素)使用;也可当成双端队列使用,即可以在队列的两端(头或尾)将元素加入或移除。
单次加入/移除元素的平均时间复杂度均为O(1)。

那么问题来了,在用作栈时到底用ArrayDeque好还是LinkedList好呢?

注意到ArrayDeque源码注释中有一句话:
This class is likely to be faster than {@link Stack} when used as a stack,
and faster than {@link LinkedList} when used as a queue.

ArrayDeque用作栈时比Stack快没有疑问,用作队列的时候似乎也会比LinkedList快!

笔者经过50W数据量的测试,发现两者性能基本接近,ArrayDeque平均耗时在18-24ms,LinkedList耗时平均在20-28ms。

如果数据量上升到100W的话,ArrayDeque的优势会更明显。

结论:ArrayDeque会略胜一筹,不过差别通常可以忽略

public static void main(String[] args) {
    int length = 500000;
    int max = length;
    // 生成一个长度为length,值从1~max的随机数组
    int[] data = new RandomIntArray(length,1,length,max).next();
    int loopCount = 10;
    long t1, t2;

    t1 = System.currentTimeMillis();
    for (int i = 0; i < loopCount; i++) {
        // testArrayDeque(data);
        testLinkedList(data);
    }
    t2 = System.currentTimeMillis();

    // 测试loopCount次取平均结果
    System.out.println("timeTaken: " + String.format("%.1f", (t2-t1)/(double)loopCount));
}

public static void testArrayDeque(int[] data) {
    int length = data.length;
    Deque<Integer> stack = new ArrayDeque<>();
    for (int i = 0; i < length/2; i++) {
        stack.push(data[i]);
        stack.push(data[i+1]);
        stack.pop();
        stack.push(stack.peek()+1);
    }
}

public static void testLinkedList(int[] data) {
    int length = data.length;
    Deque<Integer> stack = new LinkedList<>();
    for (int i = 0; i < length/2; i++) {
        stack.push(data[i]);
        stack.push(data[i+1]);
        stack.pop();
        stack.push(stack.peek()+1);
    }
}

结论

ArrayDeque会略胜一筹,不过差别通常可以忽略
经过性能对比,笔者更倾向于使用ArrayDeque来表达Java中的栈功能。


API积累

Deque中常用方法:

以这2个为基础整出来的Deque除了结构不一样,方法都一样的。

把Deque当栈用的时候:
入栈push(E e)
出栈poll() / pop() 后者在栈空的时候会抛出异常,前者返回null
查看栈顶peek() 为空时返回null
把Deque当队列用的时候:
入队offer(E e)
出队poll() 为空时返回null
查看队首peek() 为空时返回null

有些时候需要进行一些骚操作的时候(比如取得栈底元素,取得队尾元素),这些常规操作就不能满足了。
下面就是Deque中一些更详细的方法。

从上面(头部)插入:
方法名作用
void addFirst(E e)将指定的元素插入此双端队列的前面 ,空间不足抛异常
boolean offerFirst(E e)将指定的元素插入此双端队列的前面 ,空间不足插入失败返回回false
void push(E e)将指定的元素插入此双端队列的前面 ,空间不足抛异常
从上面(头部)出来/观察:
方法名作用
E removeFirst()检索并删除第一个元素,为空时抛出异常
E remove()和removeFirst一样 检索并删除第一个元素,为空时抛出异常
E pop()和removeFirst一样 检索并删除第一个元素,为空时抛出异常
E pollFirst()检索并删除第一个元素 ,为空时返回null
E poll()和pollFirst一样 检索并删除第一个元素 ,为空时返回null
E getFirst()只看看第一个元素 ,不出来,为空就抛异常
E element()和getFirst一样 只看看第一个元素 ,不出来,为空就抛异常
E peekFirst()只看看第一个元素 ,不出来,为空时返回null
E peek()和peekFirst一样 只看看第一个元素 ,不出来,为空时返回null
从下面(尾部)插入:
方法名作用
void addLast(E e)将指定的元素插入此双端队列的后面 ,空间不足抛异常
boolean offerLast(E e)将指定的元素插入此双端队列的后面,空间不足返回false
boolean add(E e)将指定的元素插入此双端队列的后面,空间不足抛异常
boolean offer(E e)将指定的元素插入此双端队列的后面,空间不足返回false
从下面(尾部)出来/观察:
方法名作用
E removeLast()检索并删除最后一个元素,为空时抛出异常
E pollLast()检索并删除最后一个元素 ,为空时返回null
E getLast()只看看最后一个元素 ,不出来,为空就抛异常
E peekLast()只看看最后一个元素 ,不出来,为空时返回null

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1068174.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

互联网Java工程师面试题·MySQL 篇·第一弹

目录 1、MySQL 中有哪几种锁&#xff1f; 2、MySQL 中有哪些不同的表格&#xff1f; 3、简述在 MySQL 数据库中 MyISAM 和 InnoDB 的区别 4、MySQL 中 InnoDB 支持的四种事务隔离级别名称&#xff0c;以及逐级之间的区别&#xff1f; 5、CHAR 和 VARCHAR 的区别&#xff1…

吃鸡技能全终极攻略!分享顶级干货,助您稳坐吃鸡王者宝座!

在绝地求生的游戏世界里&#xff0c;只有真正的高手才能立于不败之地。今天&#xff0c;我作为专业吃鸡行家&#xff0c;将为大家揭秘一些提高游戏战斗力的秘诀&#xff0c;并分享顶级游戏作战干货&#xff0c;让你成为绝地求生的大神&#xff01; 首先&#xff0c;让我们了解一…

【AntDesign】多环境配置和启动

环境分类&#xff0c;可以分为 本地环境、测试环境、生产环境等&#xff0c;通过对不同环境配置内容&#xff0c;来实现对不同环境做不同的事情。 AntDesign 项目&#xff0c;通过 config.xxx.ts 添加不同的后缀来区分配置文件&#xff0c;启动时候通过后缀启动即可。 config…

Spring Cloud Gateway2之路由详解

Spring Cloud Gateway路由 文章目录 1. 前言2. Gateway路由的基本概念3. 三种路由1. 静态路由2. 动态路由1. 利用外部存储2. API动态路由 3. 服务发现路由(自动路由)3.1. 配置方式3.2 自动路由&#xff08;服务发现&#xff09;原理核心源码GatewayDiscoveryClientAutoConfigur…

【轻松玩转MacOS】系统设置篇

引言 作为一个MacOS新用户&#xff0c;你是否对系统设置感到迷茫&#xff1f;是否想要定制出一个完全属于自己的MacBook&#xff1f;别担心&#xff0c;本文将带你一步步走进系统设置的世界&#xff0c;让你轻松定制出一个独一无二的MacBook。让我们开始吧&#xff01;今天&am…

开发做前端好还是后端好?这是个问题!

前言 随着互联网的快速发展&#xff0c;越来越多的人选择从事Web开发行业&#xff0c;而Web开发涉及到前端和后端两个方面&#xff0c;相信许多人都曾经对这两个方面进行过探究。而且编程世界就像一座大城市&#xff0c;前端开发和后端开发就像城市的两个不同街区。作为初学者&…

【C语言初阶】分支语句和循环语句

目录 一、什么是语句 二、分支语句 2.1 if 语句 2.1.1 悬空else 2.1.2 if 书写形式的对比 2.2 switch语句 2.2.1 在switch语句中的 break 2.2.2 default子句 2.2.3 练习 三、循环语句 3.1 while循环 3.1.1 while语句中的break 3.1.2 while语句中的continue 3.2 fo…

AndroidStudio添加一个虚拟设备

虚拟设备管理器 这个是AndroidStudio的启动界面 虚拟设备管理界面 点击加号&#xff0c;新增 选择手机型号 选择系统版本 虚拟设备名 完成

应用安全系列之三十九:JWT 相关安全问题以及最佳实践

JWT 简介 JWT是JSON Web Token 的简称,根据https://www.rfc-editor.org/rfc/rfc7519的定义如下: A string representing a set of claims as a JSON object that is encoded in a JWS or JWE, enabling the claims to be digitally signed or MACed and/or encrypted. 翻译…

滴滴发布十一大数据:延边出行需求上涨280% 西部省份成旅游热点

今年十一假期适逢中秋佳节&#xff0c;在亲友团聚和长假出游的多重期盼下&#xff0c;超级黄金周展现强劲内需&#xff0c;带动多样化的消费趋势&#xff0c;出行热情也随之高涨。滴滴出行数据显示&#xff0c;打车需求相比去年同期上涨80%&#xff0c;高峰时段每分钟呼叫突破1…

[论文工具] LaTeX论文SVG和EPS矢量图转换方法详解

祝大家中秋国庆双节快乐&#xff01; 回过头来&#xff0c;我们在编程过程中&#xff0c;经常会遇到各种各样的问题。然而&#xff0c;很多问题都无法解决&#xff0c;网上夹杂着各种冗余的回答&#xff0c;也缺乏系统的实战技巧归纳。为更好地从事科学研究和编程学习&#xff…

C语言中文网 - Shell脚本 - 1

Shell 既是一个连接用户和 Linux 内核的程序&#xff0c;又是一门管理 Linux 系统的脚本语言。Shell 脚本虽然没有 C、Python、Java、C# 等编程语言强大&#xff0c;但也支持了基本的编程元素。 第1章 Shell基础&#xff08;开胃菜&#xff09; 欢迎来到 Linux Shell 的世界&am…

吃鸡攻略大揭秘!提升战斗力,分享干货!

大家好&#xff01;我是你们的吃鸡玩家小编。今天我要和大家分享一些关于提高游戏战斗力和分享顶级游戏干货的干货&#xff01; 首先&#xff0c;我们要提到的是绝地求生作图工具推荐。作为一名吃鸡玩家&#xff0c;你一定想要在游戏中获得更多的优势。绝地求生作图工具是你必备…

HarmonyOS学习路之方舟开发框架—学习ArkTS语言(状态管理 八)

其他状态管理概述 除了前面章节提到的组件状态管理和应用状态管理&#xff0c;ArkTS还提供了Watch和$$来为开发者提供更多功能&#xff1a; Watch用于监听状态变量的变化。$$运算符&#xff1a;给内置组件提供TS变量的引用&#xff0c;使得TS变量和内置组件的内部状态保持同步…

WSL 安装 NVIDIA显卡驱动

文章目录 WSL 安装 NVIDIA显卡驱动本机显卡信息验证安装 WSL 版 Ubuntu 22.04在 WSL 中安装 NVIDIA显卡驱动WSL 安装 NVIDIA显卡驱动 最近在研究一些 AIGC 工具,由于 Windows 加入了 WSL 之后的各种特性,本文记录一下如何在 WSL 的 Linux发行版 中安装 NVIDIA 显卡驱动的步骤,…

力扣第110题 平衡二叉数 c++ 树 深度优先搜索 二叉树

题目 110. 平衡二叉树 简单 给定一个二叉树&#xff0c;判断它是否是高度平衡的二叉树。 本题中&#xff0c;一棵高度平衡二叉树定义为&#xff1a; 一个二叉树每个节点 的左右两个子树的高度差的绝对值不超过 1 。 示例 1&#xff1a; 输入&#xff1a;root [3,9,20,null…

车险计算器微信小程序源码 带流量主功能

车险计算器微信小程序源码带流量主功能&#xff0c;可以精准的算出车险的书目&#xff0c;是一个非常实用的微信小程序源码。 简单的计算让你得知车险价值 另外也支持流量主&#xff0c;具体小编也就不多说了&#xff0c;大家自己搭建研究吧。 源码下载&#xff1a;https://d…

nsoftware Cloud SMS 2022 .NET 22.0.8 Crack

nsoftware Cloud SMS 能够通过各种流行的消息服务&#xff08;包括 Twilio、Sinch、SMSGlobal、SMS.to、Vonage、Clickatell 等&#xff09;发送、接收和安排 SMS 消息&#xff0c;从而提供了一种简化且高效的消息服务方法。 Cloud SMS 提供单个 SMS 组件&#xff0c;允许通过…

spring源码解析——IOC之bean创建

正文 在 Spring 中存在着不同的 scope&#xff0c;默认是 singleton &#xff0c;还有 prototype、request 等等其他的 scope&#xff0c;他们的初始化步骤是怎样的呢&#xff1f;这个答案在这篇博客中给出。 singleton Spring 的 scope 默认为 singleton&#xff0c;第一部…

C++11(左值(引用),右值(引用),移动语义,完美转发)

目录 一、左值与左值引用 1、左值 2、左值引用 3、意义 二、右值与右值引用 1、右值 2、右值引用 三、右值引用使用场景和意义 1、右值的分类 2、移动构造 3、移动赋值 四、万能引用 五、完美转发 一、左值与左值引用 1、左值 左值是一个表示数据的表达式(如变量名…