常见的List接口的实现类

2023-11-05

常见的List接口的实现类

  • ArrayList:数组实现,查询快,增删慢,轻量级;(线程不安全)
  • LinkedList:双向链表实现,增删快,查询慢 (线程不安全)
  • Vector:数组实现,重量级 (线程安全、使用少)

ArrayList实现类

public class ArrayList<E> extends AbstractList<E> implements List<E>, RandomAccess, Cloneable, java.io.Serializable

内部实现

transient Object[] elementData;  用于存储数据,体现ArrayList采用的是数组的方式提供实现

构造器

//new ArrayList(1000);
public ArrayList(int initialCapacity) {  //参数是初始化容积
	if (initialCapacity > 0) { 如果容积初始值大于0则创建对应的对象
    	this.elementData = new Object[initialCapacity];
    } else if (initialCapacity == 0) { 如果容积值为0则创建一个空数组
        this.elementData = EMPTY_ELEMENTDATA;
    } else { 如果小于0则抛出一个运行时异常
    	throw new IllegalArgumentException("Illegal Capacity: "+ initialCapacity);
    }
}
//new ArrayList();
 public ArrayList() {  没有初始化参数值,则自动创建一个0个长的空数组
        this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
}

add方法的实现

//向存储数据的elementData添加新元素
public boolean add(E e) {
	ensureCapacityInternal(size + 1);  //确保内部容量,处理数组elementData的长度,确保可以存放数据,如果当前数组长度不足,则需要增加数组长度。参数的含义是满足条件的最小容积
	elementData[size++] = e;
	return true;   //如果添加过程中不出异常,则返回一定是true
}

ensureCapacityInternal方法

private void ensureCapacityInternal(int minCapacity) {
        ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}
    
private void ensureExplicitCapacity(int minCapacity) {
	modCount++;  //修改次数加1
	if (minCapacity - elementData.length > 0)  如果所需要的最小容积大于实际存储数据的数组长度,则需要进行扩容处理
		grow(minCapacity);
}
    
private void grow(int minCapacity) {
	int oldCapacity = elementData.length;  //获取实际数组的长度
	//符号>>表示二进制向右移位计算,>>1表示除以2,>>2表示除以4(2的平方).如果<<表示乘以2的多少次方
	int newCapacity = oldCapacity + (oldCapacity >> 1);  //新的容积值=旧有容器*1.5
	if (newCapacity - minCapacity < 0)  新容器如果小于所需要的最小容积,则新容积为最小容积值
		newCapacity = minCapacity;
            
	//int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8
	if (newCapacity - MAX_ARRAY_SIZE > 0)   如果新容积值大于所允许的最大容积值
	    newCapacity = hugeCapacity(minCapacity);  获取满足最小容积需求的合法的容积值
	//从elementData拷贝所有的数组元素,Arrays是工具类,提供了常见的针对数组的操作方法
	elementData = Arrays.copyOf(elementData, newCapacity);
}
    
private static int hugeCapacity(int minCapacity) {
    if (minCapacity < 0)   如果最小容积值小于0则抛出错误,表示OOM内存溢出
        throw new OutOfMemoryError();
        //例如获取的最小容积值为Integer.MAX_VALUE-7
    return (minCapacity > MAX_ARRAY_SIZE) ? Integer.MAX_VALUE :
        MAX_ARRAY_SIZE;
}

add方法用于向集合中添加元素,如果ArrayList中真正存放数据的数组长度不足,则新建一个数组,新数组的长度为原始长度1.5倍,并拷贝原始数组中的数据到新数组中,最后再追加新元素

LinkedList

类定义

public class LinkedList<E> extends AbstractSequentialList<E> implements List<E>, Deque<E>,
Cloneable, java.io.Serializable
//Deque是队列接口,提供一个双端队列的访问方法实现

底层实现为双向链表

private static class Node<E> { //节点定义
	E item; //具体存储的数据
	Node<E> next; //向后的指针
	Node<E> prev; //向前的指针
}

LinkedList类中的数据存储

transient Node<E> first; //头指针,指向链表的第一个元素
transient Node<E> last; //尾指针,指向链表的最后一个元素

对应的构造器

public LinkedList() { //没有初始化容积
}

add方法中

  • 创建Node对象,其中包含需要添加的元素值
  • Node对象的next为null
  • prev指向last
  • last对象的next为新建对象Node
  • last指向新建的Node对象

Vector

类定义
属于老版本提供的,从1.0,而ArrayList比较新,从1.2。属于线程安全的类,大部分方法上都有synchronized,一般用于要求线程安全的属性定义

public class Vector<E> extends AbstractList<E> implements List<E>, RandomAccess, Cloneable,
java.io.Serializable

数据存储

protected Object[] elementData; 采用也是数组的方式存储数据

构造器方法

public Vector() {
	this(10); //表示调用当前类的其它构造器,初始化容积为10,增长的步长值为0 
}
public Vector(int initialCapacity) {
	this(initialCapacity, 0);
}
public Vector(int initialCapacity, int capacityIncrement) { //参数1是初始化容积,参数2是容
积增长的步长值
	super();
	if (initialCapacity < 0)
		throw new IllegalArgumentException("Illegal Capacity: "+ initialCapacity);
	this.elementData = new Object[initialCapacity]//按照初始化容积值构建对应的数组
	this.capacityIncrement = capacityIncrement; 
}

add新增元素的方法实现

public synchronized boolean add(E e) {//线程安全的方法
	modCount++; //修改次数+1
 	ensureCapacityHelper(elementCount + 1); //处理容积
 	elementData[elementCount++] = e; //在数组中存储元素
 	return true; 
 }
private void ensureCapacityHelper(int minCapacity) {
	// overflow-conscious code
 	if (minCapacity - elementData.length > 0) //需要储存的数据超出数组可以存放的数据格
式,则需要进行增长
 		grow(minCapacity);
}
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) { //和ArrayList处理一致
 	if (minCapacity < 0) //OOM 内存溢出
 		throw new OutOfMemoryError();
 	return (minCapacity > MAX_ARRAY_SIZE) ?
 	Integer.MAX_VALUE :
 	MAX_ARRAY_SIZE;
}
  • 方法同步
  • 增长为2倍
  • 无参数创建时是10个长的数组

删除元素的方法remove(Object)

public boolean remove(Object o) {
	return removeElement(o);
}
public synchronized boolean removeElement(Object obj) {
	modCount++;
	int i = indexOf(obj); //查找元素的索引值
	if (i >= 0) {
		removeElementAt(i); //按照索引删除指定元素
		return true;
	}
	return false;
}
public synchronized void removeElementAt(int index) {
	modCount++;
	//针对index索引值进行合法性验证
	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);
	}
	elementCount--; //元素个数-1
	elementData[elementCount] = null; //将数组末尾的元素值赋null
}

List总结

ArrayList LinkedList Vector
实现方式 数组,按照索引下标访问速度快O(1),但是当删除、添加元素时会导致元素的移动,速度慢O(n) 双向链表,按照索引下标访问速度慢O(n),但是删除添加元素速度快O(1) 数组,按照索引下标访问速度快O(1),但是当删除添加元素时会导致元素的移动,速度慢O(n)
是否同步 不同步,线程不安全,但是并发高,访问效率高 不同步,线程不安全,但是并发高,访问效率高 同步,所以线程安全,但是并发低,访问效率低
如何选择 经常需要快速访问,较少在中间增加删除元素时使用;如果多线程访问,则需要自行编程解决线程安全问题 经常需要在内部增删元素,但是很少需要通过索引快速访问时使用;如果多线程访问,则需要自行编程解决线程安全问题 一般不使用,如果在多线程访问时可以考虑使用
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

常见的List接口的实现类 的相关文章

随机推荐

  • 自举电路原理分析

    原文来自公众号 工程师看海 自举电路字面意思是自己把自己抬起来的电路 是利用自举升压电容的升压电路 是电子电路中常见的电路之一 我们经常在IC外围器件中看到自举电容 比如下图同步降压转换器 BUCK 电路中 Cboot就是自举电容 为什么要
  • DataTables从入门到精通

    DataTables从入门到精通
  • ListView与适配器

    ListView与适配器 ListView是什么 ListView是一个以垂直方式在项目中显示视图的列表 即在一个窗口里可以滚动查看数据 比如说查看聊天记录 是一种不能实现确定视图中的内容的适配器视图 adapter view 数据和视图的
  • $nextTick实现原理详解

    vue中有一个较为特殊的API nextTick 根据官方文档的解释 它可以在DOM更新完毕之后执行一个回调 用法如下 修改数据 vm msg Hello DOM 还没有更新 Vue nextTick function DOM 更新了 复制
  • 5 个免费开源的 3D 建模/渲染工具。

    5 个开源 3D 建模 渲染工具 3八 2011 作者 riku 本文采用 CC BY NC SA 2 5协议授权 转载请注明 本文链接 5 个免费开源的 3D 建模 渲染工具 1 Art of Illusion 跨平台 支持 Window
  • 15000cd是多少流明_光通量(lm)发光强度(cd)照度单位(lux)之间的关系

    光通量 lm 发光强度 cd 照度单位 lux 之间的关系 光通量 lm 由于人眼对不同波长的电磁波具有不同的灵敏度 我们不能直接用光源的辐 射功率或辐射通量来衡量光能量 必须采用以人眼对光的感觉量为基准的单位 光通量来衡量 光通量的用符号
  • SetUnhandledExceptionFilter处理未捕获异常

    一 首先看下百度上的对此函数的解释 设置异常捕获函数 当异常没有处理的时候 系统就会调用SetUnhandledExceptionFilter所设置异常处理函数 例如一些程序在出错的时候 会向用户报告说程序那出错就是利用这个 例如QQ 二
  • github时好时坏连接不上的问题

    1 找到自己的hosts文件 直接百度 hosts文件地址 一般都是C Windows System32 drivers etc 2 用ip在线查询工具查询github网站的ip地址 3 用记事本打开hosts文件 如图添加内容 我下载有的
  • 【Python】本地版 Whisper 自动转录器(附源码网址)

    目 录 一 实时自动语音转录器简介 二 开源Whisper实时转录器 三 pyinstaller 打包发布exe应用程序 四 修改版源代码 一 实时自动语音转录器简介 实时自动语音转录器是一种能够自动将语音信号转换为文字的应用程序 它通常具
  • 服务器被攻击怎么办?如何防止服务器被攻击?

    目前 服务器遭受攻击已经成为屡见不鲜的事情了 而且大部分企业都发生过服务器被攻击的情况 从而导致业务无法正常运行 造成严重的损失和影响 那么服务器被攻击怎么办 如何有效应对服务器被攻击呢 跟着小编来看看吧 1 换高防IP或切换高防服务器 流
  • 【华为OD机试真题 Java】创建二叉树

    前言 本专栏将持续更新华为OD机试题目 并进行详细的分析与解答 包含完整的代码实现 希望可以帮助到正在努力的你 关于OD机试流程 面经 面试指导等 如有任何疑问 欢迎联系我 wechat steven moda email nansun09
  • Binder机制详解(二)

    系列章节 Binder机制详解 一 Binder机制详解 三 文章目录 前言 一 什么是MMU 二 发展历史 三 相关概念 四 分页机制 1 页表的概念 2 页式内存管理 总结 前言 上一章通过一个例子让我们认识了Binder通信机制不同于
  • HbuilderX微信小程序uniapp分包小白教程&趟坑【伸手党福利】【干货】

    本教程为小白教程 主管操作 具体原理讲解欢迎评论区补充 微信小程序分包原因 1 多人开发 2 引入了大型js 3 单项目多模块需要分包 官方资料 https developers weixin qq com miniprogram dev
  • 扫描指定路径下有多少行代码

    import java io BufferedReader import java io File import java io FileReader import java io IOException Created by qiaoju
  • 使用蓝牙耳机听群晖ds218play中的音乐(audio station)

    缘起 有时需要欣赏nas中的音乐而又不影响家人 有什么方法呢 思路 研究了一下 发现新版的群晖dms支持蓝牙usb蓝牙适配器 可以使用audio station播放 蓝牙耳机收听 步骤 1 购买CSR USB蓝牙适配器 2 插入ds218p
  • 大数据CDC技术

    1 简介 CDC全称是Change Data Capture 是一种捕获增量数据的技术统称 目前主要应用在捕获数据库数据变更的技术 其中数据库变更包括DDL DML DCL等语句触发的变更 在数据备份容灾 数据分发 面向数仓的数据集成等场景
  • JavaScript实现WebService的http的Post请求

    javascript 这个脚本实现Webservice调用 function AjaxFunc var url http localhost MyService Service asmx var method DollarConvertTo
  • 使用Jmeter做压力测试,参数化

    1 首先在工作台下添加一个线程组 测试计划右键 添加 线程 用户 线程组 根据需求填写线程组信息 根据测试数据量填写 线程数也就是并发数 下面的调度时间代表规定的时间内完成并发 2 添加HTTP请求 在线程组下右键 添加 取样器 HTTP请
  • 微信小程序image组件的mode总结+介绍(包含heightFix)

    2 10 3版本后 微信小程序的图片即image组件新增了heightFix属性 mode 总共具有14种属性 满足各种情况的放置需要 14种属性可以分为两大类 一种是完全保留的缩放属性 一种是裁剪属性 原图 缩放属性 scaleToFil
  • 常见的List接口的实现类

    常见的List接口的实现类 ArrayList 数组实现 查询快 增删慢 轻量级 线程不安全 LinkedList 双向链表实现 增删快 查询慢 线程不安全 Vector 数组实现 重量级 线程安全 使用少 ArrayList实现类 pub