最稳妥方式是用 Collection.toArray(T[]) 方法,传入 new T[0] 空数组,避免 ClassCastException;因无参 toArray() 恒返 Object[],强转 String[] 等会失败;原始类型数组需流式映射。

Java 中将集合转数组最稳妥的方式是用 Collection.toArray(T[]) 方法,而不是无参的 toArray() —— 后者返回 Object[],强转容易触发 ClassCastException,尤其在泛型集合上极易出错。
为什么 toArray() 会抛 ClassCastException
无参 toArray() 总是返回 Object[],即使你声明的是 List。JVM 不会在运行时保留泛型类型信息,所以编译器无法帮你做安全转换。
- 错误写法:
List
list = Arrays.asList("a", "b"); String[] arr = (String[]) list.toArray(); // 运行时报 ClassCastException - 根本原因:返回的是
Object[],不是String[],JVM 禁止跨数组类型强制转型 - 哪怕内容全是
String,Object[]和String[]是不同运行时类型,不能直接强转
正确用法:传入带类型的空数组(推荐)
调用 toArray(T[]) 并传入一个长度为 0 的目标类型数组,让 JDK 内部通过反射创建正确类型的数组。
- 标准写法:
List
list = Arrays.asList("x", "y", "z"); String[] arr = list.toArray(new String[0]); // ✅ 安全、简洁、JDK 1.5+ - 传
new String[0]比new String[list.size()]更优:避免预估大小失误,且现代 JVM 对零长数组优化很好 - 如果传入的数组足够大(≥ 集合 size),JDK 直接复用该数组;否则新建一个——所以传
[0]不影响性能
注意 ArrayList 和 LinkedList 的细微差异
虽然接口统一,但底层实现会影响数组填充行为(尤其当传入非空数组时):
立即学习“Java免费学习笔记(深入)”;
-
ArrayList.toArray(T[]):若传入数组长度 ≥ size,会把元素复制进该数组,并将索引size处置为null(用于标记结束) -
LinkedList.toArray(T[]):不置null,只填满前size个位置,其余保持原值(可能造成脏数据) - 因此,**永远不要依赖传入数组的“剩余位置”是否被清空**,一律用
new T[0]避免歧义
特殊场景:原始类型数组(如 int[])不能直接转
Java 集合只能存引用类型,List 无法用 toArray 直接得到 int[] —— 这是类型系统限制,不是 API 缺陷。
- 必须手动映射:
List
list = Arrays.asList(1, 2, 3); int[] arr = list.stream().mapToInt(Integer::intValue).toArray(); // ✅ JDK 8+ - 第三方库如 Guava 提供
Ints.toArray(list),但本质也是遍历+复制 - 别试图用
list.toArray(new int[0]):编译不通过,因为int不是引用类型,不满足T[]泛型约束
真正容易被忽略的点在于:很多人记住了“用带参 toArray”,却仍传 new String[list.size()],以为更高效;实际上零长数组语义更清晰、行为更确定,且 JIT 已经抹平了那点分配开销。别为过早优化牺牲可读性和健壮性。










