最安全通用的方式是用 Collection.toArray(T[]) 方法;无参 toArray() 返回 Object[],强转泛型数组会因泛型擦除和JVM类型检查导致 ClassCastException,正确写法为 list.toArray(new String[0])。

Java里将集合转为数组,最安全、通用的方式是用 Collection.toArray(T[]) 方法;直接调用无参 toArray() 会返回 Object[],强转泛型数组会触发运行时异常。
为什么不能用 toArray() 直接转成 String[] 这类泛型数组?
因为无参 toArray() 总是返回 Object[],而 Java 泛型擦除后无法在运行时创建真实泛型数组。下面这段代码编译通过但运行报错:
Listlist = Arrays.asList("a", "b"); String[] arr = (String[]) list.toArray(); // ClassCastException!
根本原因是 Object[] 和 String[] 是不同运行时类型,JVM 禁止向下强转数组引用。
正确写法:用 toArray(new T[0])
传入一个长度为 0 的目标类型数组,让 JVM 内部通过反射新建正确类型的数组。这是 JDK 1.5+ 推荐做法,简洁且类型安全:
立即学习“Java免费学习笔记(深入)”;
-
list.toArray(new String[0])→ 返回String[],不是Object[] - 传
new String[0]比new String[list.size()]更高效(HotSpot 对零长数组有优化) - 即使集合为空,也返回正确类型的空数组,不会为 null
注意 ArrayList 和 Arrays.asList() 的特殊行为
这两个常见来源容易踩坑:
-
Arrays.asList("a","b")返回的是固定大小的内部列表,其toArray()行为正常,但若后续修改原数组会影响该列表(它是数组的包装) -
ArrayList.toArray(new X[0])在 JDK 11+ 中已优化为直接分配新数组,无需担心性能;但旧版本中若传入过大的数组(如new X[1000]),JVM 仍会新建合适大小的数组并复制,所以始终推荐传new X[0] -
LinkedList的toArray实现需遍历两次(先算 size 再填值),比ArrayList略慢,但对多数场景可忽略
如果必须用原始类型数组(如 int[])怎么办?
Java 集合不支持原始类型,List 转 int[] 需手动映射:
Listlist = Arrays.asList(1, 2, 3); int[] arr = list.stream().mapToInt(Integer::intValue).toArray();
这是目前最直观的方式。不要试图用 list.toArray(new int[0])——编译都通不过,因为 int 不是引用类型,无法作为泛型参数或数组组件类型传给 toArray。
真正容易被忽略的是:泛型集合转数组这件事,表面看只是方法调用,背后牵扯泛型擦除、数组协变性、JVM 类型检查三重机制。哪怕只写一行 toArray(new X[0]),也要清楚它绕过了哪些限制、又依赖了哪些 JVM 特性。










