胖胖的枫叶
主页
展示
博客
产品设计
企业架构
全栈开发
效率工具
数据分析
项目管理
方法论
面试
  • openJdk-docs
  • spring-projects-docs
  • mysql-docs
  • redis-commands
  • redis-projects
  • apache-rocketmq
  • docker-docs
  • mybatis-docs
  • netty-docs
  • journaldev
  • geeksforgeeks
  • 后端进阶
  • 并发编程网
  • 英语肌肉记忆锻炼软件
  • 墨菲安全
  • Redisson-docs
  • jmh-Visual
  • 美团技术
  • MavenSearch
主页
展示
博客
产品设计
企业架构
全栈开发
效率工具
数据分析
项目管理
方法论
面试
  • openJdk-docs
  • spring-projects-docs
  • mysql-docs
  • redis-commands
  • redis-projects
  • apache-rocketmq
  • docker-docs
  • mybatis-docs
  • netty-docs
  • journaldev
  • geeksforgeeks
  • 后端进阶
  • 并发编程网
  • 英语肌肉记忆锻炼软件
  • 墨菲安全
  • Redisson-docs
  • jmh-Visual
  • 美团技术
  • MavenSearch
  • 标签索引
  • 2024年

    • 勇闯大A,宏观美债与美元周期
    • 勇闯大A,自我管理
    • 勇闯大A,交易系统
    • 勇闯大A,买入卖出
    • 勇闯大A,债券市场
    • 勇闯大A,微观&宏观与周期
    • 配置Mac环境
    • 勇闯大A,内外盘&换手率
    • 业务知识会计管理
    • 业务知识会计基础
    • 业务知识什么是财务
  • 2023年

    • 泰国投资项目BOI
  • 2022年

    • 企业架构故障管理
    • 企业架构开发债务
  • 2021年

    • Python3.8 Matplotlib员工数据分析
    • Python3.8 Matplotlib IP折线图
    • Python3.8 词云 IP地址
    • Redis RediSearch
    • Rust第一个CLI程序
    • Rust所有权
    • Rust函数与控制流
    • Rust变量与数据类型
    • Rust入门
    • 企业架构分布式系统
    • 编程式权限设计
    • Java JVM优化
    • SpringBoot MyBatis 批量
    • SpringBoot 测试Mock
    • SpringBoot Redis布隆过滤器
    • CentOS7 Jenkins 部署
    • SpringBoot WebClient
    • Docker Drone 部署
    • SpringBoot MyBatis
    • SpringBoot Redisson
    • SpringBoot MyBatis 雪花算法
    • Java Netty
    • Redis 扫描
    • CentOS7 Jenkins本地部署分级
    • Mac 安装 Neo4j Jupyter
    • Mac OpenJDK11 JavaFX 环境
    • Mac 安装 Jenv
    • SpringBoot Redis 延时队列
    • SpringBoot MDC日志
    • SpringBoot 定时任务
    • CentOS7 Nginx GoAccess
    • SpringBoot MyBatis 分析
    • SpringBoot Lucene
    • 企业架构分布式锁
    • 学习技巧减少学习排斥心理
    • SpringBoot 动态数据源
    • Docker Compose SpringBoot MySQL Redis
    • SpringBoot 阻塞队列
    • Docker Compose Redis 哨兵
    • Docker Compose Redis 主从
    • 网络通信
  • 2020年

    • SpringBoot 延时队列
    • MySQL基础(四)
    • Java 雪花算法
    • Redis Geo
    • 网络通信 Tcpdump
    • Spring SPI
    • Java Zookeeper
    • SpringBoot JMH
    • 网络通信 Wireshark
    • Docker Compose Redis MySQL
    • CentOS7 Docker 部署
    • Netty 源码环境搭建
    • MySQL基础(三)
    • CentOS7 Selenium运行环境
    • CentOS7 Nginx HTTPS
    • Java JMH
    • SpringBoot 修改Tomcat版本
    • Java Eureka 钉钉通知
    • SpringBoot 错误钉钉通知
    • Java JVM
    • Git 合并提交
    • CentOS7 OpenResty 部署
  • 2019年

    • Redis CLI
    • CentOS7 Nginx 日志
    • 编程式代码风格
    • IDEA 插件
    • Skywalking 源码环境搭建
    • SpringBoot Redis 超时错误
    • 编程式 gRPC
    • Java Arthas
    • Docker Compose Redis 缓存击穿
    • Docker ElasticSearch5.6.8 部署
    • Docker Mysql5.7 部署
    • Spring Redis 字符串
    • Docker Zookeeper 部署
    • Docker Redis 部署
    • SpringBoot Dubbo
    • CentOS7 CMake 部署
    • 应用程序性能指标
    • Java Code 递归
    • CentOS7 ELK 部署
    • CentOS7 Sonarqube 部署
    • Java Selenium
    • Java JJWT JUnit4
    • Spring 源码环境搭建
    • Java JUnit4
    • Java Web JSON Token
    • 编程式 FastDFS
    • Java XPath
    • Redis基础(二)
    • Redis基础(一)
    • Java MyBatis JUnit4
    • Java MyBatis H2 JUnit4
    • MyBatis 源码环境搭建
    • Git 配置
    • Java 核心
    • Java Dubbo
    • Java JavaCollecionsFramework
    • Java Maven
    • Java MyBatis
    • Java Spring
    • Java SpringMVC
    • MySQL
    • Redis
  • 2018年

    • Java HashMap
    • Java HashSet
    • Java Code 交换值
    • Spring Upgrade SpringBoot
    • Mac 编程环境
    • Java Log4j
    • 网络通信 Modbus
    • MySQL基础(二)
    • MySQL基础(一)
    • Java Stack
    • Java Vector
    • CentOS7 RabbitMQ 部署
    • CentOS7 Redis 部署
    • CentOS7 MongoDB 部署
    • CentOS7 基础命令
    • Java Eureka Zookeeper
    • CentOS7 MySQL 部署
    • Git 分支
    • CentOS7 Java环境配置
    • Java LinkedList
    • Java ArrayList
    • Spring Annotation Aop

Java ArrayList

本章是整理知识内容,为强化知识长期更新。

List 是是一个有序的集合,是Collection的实现类之一。

概述

  • list接口是Java Collection Framework的成员。
  • 列队允许添加重复的元素。
  • 列队允许null存在。
  • 列队是从0开始,也就是列队头元素的下标是0。
  • 列队支持泛型,这样可以避免ClassCastException异常。

数组与数组列表

​ Array(数组)它是在内存中划分出一块连续的地址空间进行元素的存储,由于它直接操作内存,所以性能比其他的集合要高一点点。但是数组也有一个严重的缺陷,就是需要指定初始化大小,并且在后续的操作中不能修改数组的结构,这很不友好。

​ ArrayList可以很好的进行动态扩容,它会随之元素的不断增加而调整数组大小。它的底层是基于数据实现的,所以也集成了数组的一些特性。比如查找、修改很快,但是新增和删除比较慢。

Array数组

ArrayList底层是由数组实现的,那么就回顾下Java中数组的特性。

public class Array1Test {

    @Test
    public void test1() {
        String[] arr = new String[5];
        System.out.println("数组的长度" + arr.length);
        for (int i = 0 ; i < arr.length ; i++) {
            arr[i] = String.valueOf(i);
            System.out.print(" ");
        }
        System.out.println(arr[0].toString());
        // 打印出0 , 数组的下表是从0开始的。
        arr[6] = null;
        // 强行添加一个元素到6的位置,然后取出来抛出ArrayIndexOutOfBoundsException。数组的大小是不能修改的。
    }
}

输出:

数组的长度5
	0
java.lang.ArrayIndexOutOfBoundsException: 6

List列表

Java的List是一个有序的集合,List是Collection接口的扩展接口。

List特性

  • Java List接口是Java Collection Framework的成员。
  • 列表允许添加重复的元素。
  • 列表允许有null元素。
  • 列表索引是从0开始,和数组一样的。
  • 列表接受泛型,尽量使用它。以免出现ClassCastException

源码分析

ArrayList中的常量以及变量

// 默认初始化容器的大小
private static final int DEFAULT_CAPACITY = 10;

// 空的对象数组。
private static final Object[] EMPTY_ELEMENTDATA = {};

// 默认的对象数组。
private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {};

// 被transient修饰不会被序列化,集合中元素个数。
transient Object[] elementData; // non-private to simplify nested class access

// 容器中元素的个数
private int size;

// 容器的最大容量
private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;

创建ArrayList的实例。

@Test
public void test3(){
    List list1 = new ArrayList();
    List list2 = new ArrayList(3);
    String[] vowels = {"a","e","i","o","u"};
    // 使用Arrays工具将数组转换成列表。
    List<String> vowelsList = Arrays.asList(vowels);
    List list3 = new ArrayList(vowelsList);
	
    System.out.println(list1.toString());
    System.out.println(list2.toString());
    System.out.println(list3.toString());
}
[]
[]
[a, e, i, o, u]

这分别使用了ArrayList三个构造方法。

// list1
public ArrayList() {
    // 参数为空的构造方法,元素是空的数组。
    this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
}

 // list2
 public ArrayList(int initialCapacity) {
     // 指定容器的初始化大小。
     if (initialCapacity > 0) {
         this.elementData = new Object[initialCapacity];
     } else if (initialCapacity == 0) {
         this.elementData = EMPTY_ELEMENTDATA;
     } else {
         throw new IllegalArgumentException("Illegal Capacity: "+
                                            initialCapacity);
     }
 }
    
 // 对应list3
 public ArrayList(Collection<? extends E> c) {
    // 将一个Collection容器的元素都复制进来。
    elementData = c.toArray();
    if ((size = elementData.length) != 0) {
        // c.toArray might (incorrectly) not return Object[] (see 6260652)
        if (elementData.getClass() != Object[].class)
            elementData = Arrays.copyOf(elementData, size, Object[].class);
    } else {
        // replace with empty array.
        this.elementData = EMPTY_ELEMENTDATA;
    }
 }

这让我想起了面试题目。

@Test
public void test4(){
    List list = new ArrayList(20);
}

题目是:生成list实例需要扩容几次。实际上是0次。因为创建的时候就指定了大小。

add方法

@Test
public void test4(){
    List list = new ArrayList();
    System.out.println(list.add(1));
    list.add(0,2);
    System.out.println(list.toString());
}

输出

true
[2, 1] // 是不是很神奇,下标0的位置是2,下标1位置值是1。

带着上面的疑问我们好好看看,顺便看看经常说ArrayList到底是如何扩容的。

// 直接添加元素
public boolean add(E e) {
    // 确保内部容量足够,不够就扩容。 size 是当前容器的数量
    ensureCapacityInternal(size + 1);  // Increments modCount!! 
    // 将元素添加到数组中
    elementData[size++] = e;
    return true;
}

// 在指定索引的位置添加元素。
public void add(int index, E element) {
    rangeCheckForAdd(index); //检查索引位置是否越界。

    ensureCapacityInternal(size + 1);  // Increments modCount!!
    // 真相就在这里。
    System.arraycopy(elementData, index, elementData, index + 1,
                     size - index);
    elementData[index] = element;
    size++;
}

System.arraycopy 是一个本地静态方法,使用它可以对数组之间进行复制。

public static native void arraycopy(Object src,  int  srcPos,
                                    Object dest, int destPos,
                                    int length);
  • src:源数组;
  • srcPos:源数组要复制的起始位置; dest:目的数组; destPos:目的数组放置的起始位置;
  • length:复制的长度。

上面的操作结果就是将数组中的元素位置进行了移动,当然神奇的地方是这个方法可以自己复制自己。

// 1
private void ensureCapacityInternal(int minCapacity) {
    // 当前实际存储的数组是空的的时候
    if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
        // 最小的数字 就是容量大小。
        minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
    }

    ensureExplicitCapacity(minCapacity);
}

// 2
  private void ensureExplicitCapacity(int minCapacity) {
        modCount++;
		// 如果当前数组的长度小于 需要的最小容量大小就扩容。
        // overflow-conscious code
        if (minCapacity - elementData.length > 0)
            grow(minCapacity);
  }

// 3
  private void grow(int minCapacity) {
        // 增加最小扩容,保证元素能添加进容器。
        // overflow-conscious code
        int oldCapacity = elementData.length;
        // 位运算 , 新的容量大小 = 当前数组长度 右移动一位  就是 newCapacity = oldCapacity + (oldCapacity / 2) 取整。也就是经常说的扩容1.5倍。
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        
        if (newCapacity - minCapacity < 0)
            newCapacity = minCapacity;
    	// 如果新的容器大小 比 容器最大值还大
        if (newCapacity - MAX_ARRAY_SIZE > 0)
            newCapacity = hugeCapacity(minCapacity);
        // minCapacity is usually close to size, so this is a win:
        // 关键的时候来了,通过Arrays.copyOf复制元素并指定新的数组大小。这个方法最终还是调用了System.arraycopy方法,其它参数是由Array.copyOf预设好的。
        elementData = Arrays.copyOf(elementData, newCapacity);
    }
// 4 抛出OutOfMemoryError异常咯。内存溢出
    private static int hugeCapacity(int minCapacity) {
        if (minCapacity < 0) // overflow
            throw new OutOfMemoryError();
        return (minCapacity > MAX_ARRAY_SIZE) ?
            Integer.MAX_VALUE :
            MAX_ARRAY_SIZE;
    }

每次扩容都是通过Arrays.copyOf(elementData, newCapacity)的来实现的。

set方法

这个方法其实就是更新方法。

@Test
public void test5(){
    List list = new ArrayList();
    list.add(1);
    System.out.println(list.set(0,1));
    System.out.println(list.toString());
}

输出

1
[1]

看下源代码。

public E set(int index, E element) {
    rangeCheck(index); //检查所以是否越界,这里要注意,该下标没有值会抛出越界异常。

    E oldValue = elementData(index); // 获取下标原始的值
    elementData[index] = element;	 // 将新值复制给对应的下标
    return oldValue;				// 返回原始的值
}

remove方法

@Test
public void test6(){
    List list = new ArrayList();
    list.add("a");
    list.add("b");
    list.add("b");
    list.add("c");
    list.add("d");
    System.out.println(list.toString());
    list.remove("b"); //根据元素匹配删除
    System.out.println(list.toString());
}

输出

[a, b, b, c, d]
[a, b, c, d] //会删除匹配到的第一个元素
// 根据下标删除元素。删除成功就返回被删除的元素。
public E remove(int index) {
    rangeCheck(index); // 检查下标是否越界

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

    int numMoved = size - index - 1;
    // 根据索引删除相比对象删除只需要复制一份自己的部分元素。
    if (numMoved > 0)
        System.arraycopy(elementData, index+1, elementData, index,
                         numMoved);
    elementData[--size] = null; // clear to let GC do its work

    return oldValue;
}

// 删除元素,如果删除成功返回true
public boolean remove(Object o) {
    // 这里看到不管待删除的值是啥,都会从头遍历到尾部。直到找到合适值在删除。所以删除这个动作非常大。
    if (o == null) { 
        for (int index = 0; index < size; index++)
            if (elementData[index] == null) {
                fastRemove(index);
                return true;
            }
    } else {
        for (int index = 0; index < size; index++)
            if (o.equals(elementData[index])) {
                fastRemove(index);
                return true;
            }
    }
    return false;
}

// 删除索引上的元素
private void fastRemove(int index) {
    modCount++;
    int numMoved = size - index - 1;
    if (numMoved > 0)
        System.arraycopy(elementData, index+1, elementData, index,
                         numMoved);
    elementData[--size] = null; // clear to let GC do its work
}

这两个删除方法可以看出,动作都比较大。这种操作会慢很多。

get方法。

相比上面3种方法确实轻量级太多了。

public E get(int index) {
    rangeCheck(index);

    return elementData(index);
}

平时也就听大家说新增修改删除慢,也就访问速度比较快。看了源码之后才清晰很多。

contains方法

这也是一个重量级的方法

// 判断对象存在与当前集合中。
public boolean contains(Object o) {
    return indexOf(o) >= 0;
}

/**
 * Returns the index of the first occurrence of the specified element
 * in this list, or -1 if this list does not contain the element.
 * More formally, returns the lowest index <tt>i</tt> such that
 * <tt>(o==null&nbsp;?&nbsp;get(i)==null&nbsp;:&nbsp;o.equals(get(i)))</tt>,
 * or -1 if there is no such index.
 */
public int indexOf(Object o) {
    // 当找不到该对象的时候返回-1,找到了就返回它的下标。
    if (o == null) {
        for (int i = 0; i < size; i++)
            if (elementData[i]==null)
                return i;
    } else {
        for (int i = 0; i < size; i++)
            if (o.equals(elementData[i]))
                return i;
    }
    return -1;
}

安全删除

@Test
    public void test2(){
        //  1、创建一个二维数组
        String[] vowels = {"a","e","i","o","u"};
        //  2、将二维数组转换成列表,在转换成容器
        Collection<String> collection = new ArrayList(Arrays.asList(vowels));
        // 3、获取容器迭代器实例
        Iterator<String> itr = collection.iterator();
        // 4、逻辑判断itr 中下个元素是否存在。
        while (itr.hasNext()){
            String var = itr.next(); // 获取将要访问的下一个元素。
            // 5、尝试修改其中的一个元素。
            if(var == "a"){
                // 5、删除掉 `a`
                itr.remove();
            }else{
                System.out.print(var);
            }
        }
    }

线程安全

由于所有的方法都没有进行同步。所以很直观的认为有线程安全的问题,还是亲自测试下避免迷迷糊糊的。

场景一:多线程对一个ArrayList进行操作。

假设我们需要添加10000个元素进ArrayList

public class ArrayListTest2 {

    /**
     * 10线程对一个ArrayList进行操作。
     *
     * @param args
     */
    public static void main(String[] args) throws InterruptedException {
        List list = new ArrayList();
        ExecutorService executorService = Executors.newFixedThreadPool(20);
        for (int sum = 0 ; sum < 10 ; sum ++){
            executorService.execute(new Runnable() {
                @Override
                public void run() {
                    System.out.println("ThreadId:" + Thread.currentThread().getId() + " start!");
                    for (int i = 0 ; i < 1000 ; i++ ){
                        list.add(i);
                    }
                    System.out.println("ThreadId:" + Thread.currentThread().getId() + " stop!");

                }
            });
        }
        executorService.shutdown();
        while (!executorService.isTerminated()){
            try {
                Thread.sleep(1000*5);  // 这里睡眠下主线程,不然其它线程还没执行完就被关闭了。
            }catch (InterruptedException e){
                e.printStackTrace();
            }
        }
        System.out.println("长度" + list.size());
    }

}

输出

ThreadId:12 start!
ThreadId:11 start!
ThreadId:12 stop!
ThreadId:11 stop!
ThreadId:14 start!
ThreadId:14 stop!
ThreadId:15 start!
ThreadId:15 stop!
ThreadId:13 start!
ThreadId:16 start!
ThreadId:16 stop!
ThreadId:17 start!
ThreadId:17 stop!
ThreadId:13 stop!
ThreadId:18 start!
ThreadId:18 stop!
ThreadId:19 start!
ThreadId:19 stop!
ThreadId:20 start!
ThreadId:20 stop!
长度9950

每次的输出结构都不同,偶尔还会抛出越界异常。

// 直接添加元素
public boolean add(E e) {
    ensureCapacityInternal(size + 1);  // Increments modCount!! 
    elementData[size++] = e;   //应该就是在这里,在多线程的情况下很有可能出现重复添加了元素。
    return true;
}

Fail-Fast场景

fail-fast 机制在遍历一个集合时,当集合结构被修改,会抛出 Concurrent Modification Exception。

fail-fast 会在以下两种情况下抛出 Concurrent Modification Exception

(1)单线程环境

  • 集合被创建后,在遍历它的过程中修改了结构。
  • 注意 remove() 方法会让 expectModcount 和 modcount 相等,所以是不会抛出这个异常。

(2)多线程环境

  • 当一个线程在遍历这个集合,而另一个线程对这个集合的结构进行了修改。

modCount 用来记录 ArrayList 结构发生变化的次数。结构发生变化是指添加或者删除至少一个元素的所有操作,或者是调整内部数组的大小,仅仅只是设置元素的值不算结构发生变化。

在进行序列化或者迭代等操作时,需要比较操作前后 modCount 是否改变,如果改变了需要抛出 Concurrent Modification Exception。

private void writeObject(java.io.ObjectOutputStream s) throws java.io.IOException{
    // Write out element count, and any hidden stuff
    int expectedModCount = modCount;
    s.defaultWriteObject();

    // Write out size as capacity for behavioural compatibility with clone()
    s.writeInt(size);

    // Write out all elements in the proper order.
    for (int i=0; i<size; i++) {
        s.writeObject(elementData[i]);
    }

    if (modCount != expectedModCount) {
        throw new ConcurrentModificationException();
    }
}
最近更新: 2025/12/27 18:51
Contributors: 庆峰
Prev
Java LinkedList
Next
Spring Annotation Aop