当前位置: 首页 > news >正文

【ArrayList是如何扩容(ArrayList、LinkedList、与Vector的区别)】

ArrayList、LinkedList、与Vector的区别

  • 解读
    • ArrayList 是一个可改变大小的数组
    • LinkedList 是一个双向链表
    • Vector 属强同步类
  • 拓展知识面
    • ArrayList是如何扩容?
    • 如何利用List实现LRU?

解读

List主要有ArrayList、LinkedList与Vector几种实现。这三者都实现了List接口,使用方式也极其相似,主要区别在于因为实现方式的不同,所以对不同的操作就有不同的效率。

ArrayList 是一个可改变大小的数组

当,更多的元素加入到ArrayList中时,其大小将会动态的增长,内部的元素可以直接通过get与set方法进行访问,因为ArrayList本质上就是数组。

LinkedList 是一个双向链表

在添加和删除元素的时间具有比ArrayList更好的性能,但是在get与set方面是弱于ArrayList。当然了,这些对比都是指数据量很大或者操作很频繁的情况下的对比,如果数据量和运算量很小,那么就没有对比的意义。

Vector 属强同步类

Vector和ArrayList类似,但它属于强同步类。如果你的程序本身是线程安全的(thread-safe,没有在多个线程之间共享同一个集合或者对象),那么使用ArrayList是更好的选择。
Vector和ArrayList在更多元素添加进来时会请求更大的空间。Vector每次请求是它自身大小的双倍空间,而ArrayList每次对size增长50%。
而LinkedList还实现了Queue个Deque接口,该接口与List相比,其提供了更多的方法,包括offer()、peek()、poll()等。

注意:默认情况下ArrayList的初始容量非常小,所以如果可以预估数据量的话,分配一个较大的初始值才属于最佳操作,这样可以减少调整大小的开销。

拓展知识面

ArrayList是如何扩容?

首先,我们要明白ArrayList是基于数组的,这个上面讲到了,我们都知道,申请数组的时间,只能申请一个定长的数组,那么List是如何实现数组扩容的呢???ArrayList的扩容有几个步骤:

  1. 检查新增元素后是否会超过数组的容量,若超过,则进行下一步扩容;
  2. 设置新增容量为原始(旧/老)容量的1.5倍,最多不超过2^31-1(在Java8中ArrayList的容量最大是Integer.MAX_VALUE-8,这是由于在Java8中,ArrayList内部实现进行了一些改进,使用了一些数组复制的技巧来提高性能和内存的利用率,而这些技巧需要额外的8个元素的空间来进行优化)。
  3. 之后呢,申请一个容量为1.5倍的数组,将原有数组的元素复制到新数组中,自此,扩容完成。

如何利用List实现LRU?

LRU,即最近最少使用策略,基于时空局部性原理(最近访问的,未来也会被访问的),往往作为缓存淘汰的策略,如Redis和GuavaMap都使用了这种套胎策略。

我们可以基于LinkedList来实现LRU,因为LinkedList基于双向链表,每个节点都会记录上一个和下一个的节点,具体实现方式如下:

public class LruListCache<E> {private final int maxSize;private final LinkedList<E> list = new LinkedList<>();public LruListCache (int maxSize) {this.maxSize = maxSize;} public void add(E e) {if(list.size() < maxSize) {list.addFrist(e);}else{list.removeLast(); list.addFrist(e);}}public E get(int index) {E e = list.get(index);list.remove(e);add(e);return e;}@Overridepublic String toString() {return list.toString();}}

OK,如果不懂数组和链表的区别的话,随后我有一个专门的数据结构的专栏,会讲到栈和队列、数组和链表以及二叉树的遍历等等内容。会做详细解说。

http://www.lryc.cn/news/239657.html

相关文章:

  • STM32_3(GPIO)
  • 【技巧】PDF文件如何编辑?
  • AR道具特效制作工具
  • 鸿蒙4.0开发笔记之DevEco Studio页面操作router的pushUrl页面跳转与back返回上一页(五)
  • 20个CSS函数-释放设计创造力和响应能力
  • Dubbo从入门到上天系列第十八篇:Dubbo引入注册中心简介以及DubboAdmin简要介绍,为后续详解Dubbo各种注册中心做铺垫!
  • CentOS8安装MySQL
  • Java集合拓展01
  • 【Django使用】md文档10大模块第5期:Django数据库增删改查和Django视图
  • 在AWS VPC中运行Nagios检查时指定自定义DNS解析器的选项
  • 【uniapp】触底加载事件 onReachBottom 不生效
  • Vue3简单使用(一) --- 环境搭建
  • 陪玩圈子系统APP小程序H5,详细介绍,源码交付,支持二开!
  • 目标检测原理
  • 2、数仓理论概述与相关概念
  • YOLOv5 分类模型 OpenCV和PyTorch两者实现预处理的差异
  • 使用NPOI处理EXCEL文件:例1-关于优化的一些问题
  • 连接k8s和凌鲨
  • C语言——结构体的应用
  • 人机交互——机器人形态
  • BGP的基础知识
  • 2023.11.18 每日一题(AI自生成应用)【C++】【Python】【Java】【Go】 动态时间序列分析
  • uniapp相关记录
  • 优质猫罐头有哪些品牌?分享5款宠物店自用值得推荐的猫罐头!
  • HTML新手入门笔记整理:HTML基本标签
  • Redis高级特性和应用(发布 订阅、Stream)
  • RoCE、IB和TCP等网络的基本知识及差异对比
  • c语言-操作符详解(含优先级与结合性)
  • ubuntu安装nvm
  • opengl制作天空盒