在Java中如何使用ArrayList存储和访问元素_Java动态数组集合使用解析

ArrayList是Java中基于Object[]实现的动态数组,长度可变、仅存引用类型、提供add()/remove()/get()等方法;与普通数组相比,它默认容量为10,扩容为1.5倍,支持泛型但需显式声明。

ArrayList 是什么,和普通数组有什么区别

ArrayList 是 Java 中的动态数组实现,底层用 Object[] 存储元素,能自动扩容。它和普通数组(如 int[])最核心的区别是:长度可变、只能存引用类型(基本类型需装箱)、提供丰富的操作方法(如 add()remove()get())。

常见误区是以为 ArrayList 合法——实际会编译报错,必须写成 ArrayList;另外,初始化时不指定容量时,默认底层数组大小为 10,扩容策略是「原容量 × 1.5」(JDK 17+),不是翻倍。

如何正确创建、添加和访问元素

创建时建议显式指定泛型类型,避免运行时 ClassCastException;添加元素用 add(),访问用 get(index),注意索引从 0 开始,越界会抛 IndexOutOfBoundsException

  • add(E e):追加到末尾;add(int index, E e):在指定位置插入(会移动后续元素)
  • get(int index):仅支持随机访问,时间复杂度 O(1);但 remove(int index) 是 O(n),因为要复制数组
  • 空集合调用 get(0)remove(0) 一定抛异常,务必先检查 isEmpty()size() > 0
ArrayList list = new ArrayList<>();
list.add("hello");
list.add(0, "world"); // 插入到开头 → ["world", "

hello"] String s = list.get(1); // "hello"

遍历时该用 for 还是 foreach?要注意什么

两种方式都行,但行为不同:

  • 普通 for (int i = 0; i :适合需要索引、或边遍历边修改(如删除满足条件的元素)的场景
  • 增强 for 循环(for (String s : list)):本质调用 Iterator,安全但禁止在循环中直接调用 list.remove(),否则抛 ConcurrentModificationException
  • 真要边遍历边删,得用 Iterator.remove(),或者倒序用普通 for(避免索引错位)
Iterator it = list.iterator();
while (it.hasNext()) {
    String s = it.next();
    if (s.length() == 5) it.remove(); // 安全删除
}

性能和内存开销容易被忽略的点

ArrayList 不是万能的。频繁在头部或中间 add()/remove() 会导致大量数组复制,此时应考虑 LinkedList(但注意它不支持高效随机访问);另外,如果初始容量远大于实际使用量(比如 new ArrayList(1000) 只存 10 个元素),会浪费内存。

还有一个隐蔽问题:ArrayList 的 toArray() 方法返回的是 Object[],不能直接强转为 String[] 等具体类型数组,否则 ClassCastException;正确写法是 list.toArray(new String[0])list.toArray(String[]::new)(Java 11+)。