最安全的List转数组方式是传入类型化空数组(如list.toArray(new String[0])),避免ClassCastException;Set转数组顺序不保证,需用LinkedHashSet或TreeSet确保顺序;Arrays.asList()返回不可变列表,修改会抛异常。

用 toArray() 方法转换 List 到数组最安全
直接调用 list.toArray() 返回的是 Object[],不能强制转成具体类型数组(比如 String[]),否则会抛 ClassCastException。真正安全的方式是传入一个带类型的空数组:
Listlist = Arrays.asList("a", "b", "c"); String[] arr = list.toArray(new String[0]);
传 new String[0] 而不是 new String[list.size()] —— HotSpot 对零长度数组做了优化,避免多余内存分配;JDK 11+ 中两者性能几乎无差别,但语义更清晰。
- 传
null会触发NullPointerException - 传入数组长度小于 list 大小,会自动新建同类型数组返回
- 传入数组长度大于 list 大小,多出的元素会被设为
null(仅当数组引用被复用时)
Set 转数组也要用 toArray(T[]),但顺序不保证
HashSet 本身无序,toArray() 返回的数组元素顺序不确定;如果需要稳定顺序,先转成 LinkedHashSet 或排序后的 TreeSet 再转:
Setset = new HashSet<>(Arrays.asList(3, 1, 4)); Integer[] arr = set.toArray(new Integer[0]); // 顺序随机
注意:即使使用 toArray(new Integer[set.size()]),也无法改变底层遍历顺序,只是避免数组扩容。
立即学习“Java免费学习笔记(深入)”;
-
LinkedHashSet保持插入顺序 -
TreeSet按自然序或自定义比较器排序 - 不要依赖
HashSet.toArray()的输出顺序做逻辑判断
数组转 List 用 Arrays.asList() 有坑
Arrays.asList(arr) 返回的是固定大小的 List 实现(内部是 Arrays$ArrayList),不支持 add()、remove()、clear(),调用会抛 UnsupportedOperationException:
String[] arr = {"x", "y"};
List list = Arrays.asList(arr);
list.add("z"); // 运行时报错
若需可修改的 List,必须显式拷贝:
ListmutableList = new ArrayList<>(Arrays.asList(arr));
-
Arrays.asList()返回的 List 与原数组内容共享引用(修改对象属性会影响彼此,但替换数组元素不会同步) - 基本类型数组(如
int[])传入会变成单元素 List:Arrays.asList(new int[]{1,2})→[I@xxx - 要处理基本类型,得用 Stream:
IntStream.of(arr).boxed().collect(Collectors.toList())
泛型数组无法直接创建,绕过方式要明确
Java 类型擦除导致不能写 new T[10],所以 toArray() 必须靠运行时类型信息(即你传入的数组)来确定目标类型。这也是为什么 list.toArray()(无参)只能返回 Object[]。
如果你封装工具方法,别试图“猜”类型:
public staticT[] toTypedArray(List list, Class clazz) { @SuppressWarnings("unchecked") T[] arr = (T[]) Array.newInstance(clazz, list.size()); return list.toArray(arr); }
这种写法需要调用方传 String.class,且对泛型通配符(如 List extends Number>)依然受限。
- 不要用
(T[]) new Object[n]强转,这是不安全的泛型转型 -
ArrayList.toArray()底层就是用Arrays.copyOf()+ 反射创建数组,你不必重复造轮子 - 在 Android 或低版本 JDK 上,
new T[0]可能触发警告,但行为正确










