Java源码解读系列(一):ArrayList

本文简单介绍了 ArrayList,并对扩容,添加,删除操作的源代码做分析。能力有限,欢迎指正。

ArrayList是什么?

ArrayList 就是数组列表,主要用来装载数据。底层实现是数组 Object[] elementData,当我们装载的是基本数据类型 int, long, boolean, shot...的时候我们只能存储他们对应的包装类型。

与它类似的是 LinkedList,和 LinkedList 相比,它的查找和访问元素的速度较快,但新增,删除的速度较慢。

线程安全吗?

线程不安全。

正常使用场景中,ArrayList 都是用来查询,不会涉及太频繁的增删,如果涉及频繁的增删,可以使用 LinkedList。如果需要线程安全就使用 Vector

VectorArrayList 的线程安全版本,实现方式就是在所有方法加上synchronized,性能较差。

如何扩容?

因为数组的大小是固定,当容量超出了现有数组的大小,就需要进行扩容。

private void grow(int minCapacity) {
    // overflow-conscious code
    int oldCapacity = elementData.length;
    // 每次扩大原有容量的一半
    int newCapacity = oldCapacity + (oldCapacity >> 1);
    // 如果扩大一半后还是无法满足,则使用minCapacity
    if (newCapacity - minCapacity < 0)
        newCapacity = minCapacity;
    // 如果超过最大size,则获取最大容量的数组
    if (newCapacity - MAX_ARRAY_SIZE > 0)
        newCapacity = hugeCapacity(minCapacity);
    // minCapacity is usually close to size, so this is a win:
    elementData = Arrays.copyOf(elementData, newCapacity);
}

为什么说ArrayList插入效率低?

原因有两点:

  1. 新增就要检测容量够不够,如果不够就需要扩容
  2. 尾部新增比较快,如果是在数组头部或者中部新增就会慢很多,因为要把后面的元素全部往后移一位
  3. 把元素往后移一位使用的是复制 System.arraycopy(),它是native方法(java定义了接口,其他语言进行实现),所以比较快
/**
 * 添加在尾部
 */
public boolean add(E e) {
    // 检查容量
    ensureCapacityInternal(size + 1);
    // 添加在尾部
    elementData[size++] = e;
    return true;
}

/**
 * 按指定位置添加
 */
public void add(int index, E element) {
    // 检查index
    rangeCheckForAdd(index);

    // 检查容量
    ensureCapacityInternal(size + 1);
    // index后面的元素全部往后移一位
    System.arraycopy(elementData, index, elementData, index + 1,
                        size - index);
    elementData[index] = element;
    size++;
}

删除元素效率如何?

效率和新增差不多,都是要移动元素,但是不需要检查容量和扩容。

public E remove(int index) {
    // 检查index
    rangeCheck(index);

    modCount++;
    E oldValue = elementData(index);

    int numMoved = size - index - 1;
    if (numMoved > 0)
        // index后面的元素全部往前移一位
        System.arraycopy(elementData, index+1, elementData, index,
                            numMoved);
    elementData[--size] = null; // clear to let GC do its work

    return oldValue;
}

适合做队列吗?

非常不适合。

队列是FIFO,在尾巴进,头部出,出的时候需要移动后面所有数据,效率很低。链表比较适合做队列。

new ArrayList<>(18) 会不会初始化数组大小?

不会初始化数组大小!!!

这是Java Bug。

而且将构造函数与initialCapcity结合使用,然后使用set()方法会抛出异常。

public static void main(String[] args) {
    ArrayList<Integer> a = new ArrayList<>(12);
    System.out.println(a.size());
    a.set(3, 3);
}

总结

  1. 底层实现是数组 Object[] elementData
  2. 查找和访问元素的速度较快,但新增,删除的速度较慢
  3. 线程不安全
  4. 每次扩容原有数组大小的一半

原文地址:https://www.cnblogs.com/java1024/p/12355467.html

时间: 2024-08-24 13:16:45

Java源码解读系列(一):ArrayList的相关文章

Java源码分析系列之ArrayList读后感

1.前言 在平时的开发中,Java集合一直是比较常用的.以前,对集合的掌握并不深入,仅简单的使用增删改查的相关方法.这个星期,抽时间反复的读了几遍ArrayList的源码,有了一些收获,写出来给自己,也希望对大家有帮助. 2.走进ArrayList 看一下ArrayList的声明和属性 声明: public class ArrayList<E> extends AbstractList<E>         implements List<E>, RandomAcces

JAVA源码解读---HashMap目录扩展的奥秘

摘要:为了探索JAVA1.7源码中HashMap类数据的组织方法与目录扩展方法,本文通过对JAVA1.7源码中HashMap类源码的阅读与分析,得出结论:hashmap中存储数据的数据结构采用的是链表数组,目录是个数组,数组的成员是链表.冲突解决方法:典型的链地址法,冲突后,在链表头部插入数据.目录扩展方法:已二倍的方式扩展,一直到目录的最大上限.目录扩展的触发条件:装载因子的方式触发.从java中hashmap的实现可以看出,桶数据的组织方式并不是一种非常高效的方式.对检索效率不利.同时,数据

swoft| 源码解读系列二: 启动阶段, swoft 都干了些啥?

date: 2018-8-01 14:22:17title: swoft| 源码解读系列二: 启动阶段, swoft 都干了些啥?description: 阅读 sowft 框架源码, 了解 sowft 启动阶段的那些事儿 小伙伴刚接触 swoft 的时候会感觉 压力有点大, 更直观的说法是 难. 开发组是不赞成 难 这个说法的, swoft 的代码都是 php 实现的, 而 php 又是 世界上最好的语言, swoft 的代码阅读起来是很轻松的. 之后开发组会用 系列源码 解读文章, 深入解析

java源码解读--queue

queue接口特点:可以模拟队列行为,即"先进先出". 接口结构 queue接口继承了Collection接口,并增加了一些新方法 12345678910111213141516 public interface <E> extends Collection<E>{ boolean add(E e); //将元素插入队列,如果失败返回false boolean offer(E e); //移除并返回队列中的第一个元素,队列为空时,抛异常 E remove();

Tomcat源码解读系列(一)——server.xml文件的配置

Tomcat是J2EE开发人员最常用到的开发工具,在Java Web应用的调试开发和实际部署中,我们都可以看到Tomcat的影子.大多数时候,我们可以将Tomcat当做一个黑盒来看待,只需要将编写的Java Web工程进行部署即可,但是,在遇到一些比较复杂难解决的问题时,如果我们了解了Tomcat的内部实现原理将会处理起来更得心应手更快地定位问题.另外,通过学习Tomcat的源码还可以更加深入地了解JEE规范,学习常见的设计模式.本系列的文章,将会介绍Tomcat的核心功能是如何实现的,一方面作

Java 源码学习系列(三)——Integer

Integer 类在对象中包装了一个基本类型 int 的值.Integer 类型的对象包含一个 int 类型的字段. 此外,该类提供了多个方法,能在 int 类型和 String 类型之间互相转换,还提供了处理 int 类型时非常有用的其他一些常量和方法. 类定义 public final class Integer extends Number implements Comparable<Integer> 从类定义中我们可以知道以下几点: 1.Integer类不能被继承 2.Integer类

Java源码分析系列

1) 深入Java集合学习系列:HashMap的实现原理 2) 深入Java集合学习系列:LinkedHashMap的实现原理 3) 深入Java集合学习系列:HashSet的实现原理 4) 深入Java集合学习系列:LinkedHashSet的实现原理 5)深入Java集合学习系列:ArrayList的实现原理

springMVC 源码解读系列(一)初始化

先看看DispatcherServlet的类机构: 初始化时序图: servlet初始化会调用 init 方法,换句话说就是springMVC进行初始化的时候首先会去执行HttpServletBean的init方法, 下面看看HttpServletBean的源码: 上面这段代码主要是在获取你在web.xml中配置在<init-param>中的属性(例如: namespace, contextConfigLocation). 其中有一点值得注意,那就是 initServletBean() 这个方

Java源码分析系列之HttpServletRequest源码分析

从源码当中 我们可以 得知,HttpServletRequest其实 实际上 并 不是一个类,它只是一个标准,一个 接口而已,它的 父类是ServletRequest. 认证方式 public interface HttpServletRequest extends ServletRequest 从阅读源码 当中 ,我们 可以 获得 如下认证信息: /** * String identifier for Basic authentication. Value "BASIC" */ pu