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
ArrayListlist = new ArrayList<>(); list.add("hello"); list.add(0, "world"); // 插入到开头 → ["world", "hello"] String s = list.get(1); // "hello"
遍历时该用 for 还是 foreach?要注意什么
两种方式都行,但行为不同:
立即学习“Java免费学习笔记(深入)”;
- 普通
for (int i = 0; i :适合需要索引、或边遍历边修改(如删除满足条件的元素)的场景 - 增强 for 循环(
for (String s : list)):本质调用Iterator,安全但禁止在循环中直接调用list.remove(),否则抛ConcurrentModificationException - 真要边遍历边删,得用
Iterator.remove(),或者倒序用普通 for(避免索引错位)
Iteratorit = 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+)。










