实际上,不少开发者对Arrays.copyOf处理非基本类型数组的机制感到困惑——其本质是浅拷贝:仅复制引用地址,不会创建新的对象实例;当新数组长度超过原数组时,多余位置会自动填充null;它严格按照源数组类型创建新数组,无法自动进行类型转换或装箱操作;此外,该方法不会主动校验null或负长度,这些边界条件需要调用方自行处理。

谈及Arrays.copyOf对非基本类型数组(如String[]、Object[]或自定义对象数组)的处理,其核心特点可归纳为三个关键词:浅拷贝、默认值填充、类型安全创建。需要牢记的是,它不会对数组中的对象进行深度复制。
浅拷贝:只复制引用,不复制对象实例
copyOf 拷贝的是每个元素的内存地址(即引用),而非真正克隆出一个新对象。原数组与新数组中对应位置的元素指向同一个实例。这意味着:
- 如果修改新数组中某个元素的属性,例如执行
copy[0].setName("new"),原数组对应位置的对象也会随之改变——因为它们指向同一个内存实例。 - 然而,如果直接为新数组的某个索引重新赋值,如
copy[0] = new Person(),则原数组不会受影响——因为此时仅将新数组的引用指向了一个新创建的对象。理解这一区别至关重要。
当新长度大于原数组长度时,多余位置填充 null
当指定的新长度超过原数组长度时,多出的索引位置会被统一设置为 null——这是所有引用类型变量的默认初始值。例如:
String[] src = {"a", "b"};
String[] dst = Arrays.copyOf(src, 4);
// 结果:{"a", "b", null, null}
严格类型创建,支持泛型擦除下的安全类型转换
底层实现通过 Array.newInstance(newType.getComponentType(), newLength),根据传入数组的组件类型创建相同类型的新数组。因此:
- 传入
String[],返回的仍是String[],不会变成Object[]。 - 如需向上转型,例如将
String[]转为Object[],必须显式传递第三个参数:Arrays.copyOf(strArr, 5, Object[].class)。 - 编译时的泛型检查有效,但运行期泛型信息会被擦除,因此该方法不会自动执行装箱或拆箱操作。
空值与边界条件需要调用方自行防护
该工具方法不会主动进行空值校验。如果传入的 original 数组为 null,会直接抛出 NullPointerException;若 newLength 参数为负数,则会抛出 NegativeArraySizeException。因此,在调用前建议自行判空或校验长度参数,切勿依赖该方法进行边界保护。
