【为什么会写这篇博文?】
花花❀看到 Arrays 源码中的 copyOf 方法,底层调用的是 System 中 arraycopy 方法,而且我已经掉过不止一次数组复制的坑🕳,今天就打算写个博客给我长个记性。
Arrays是一个工具类,该类中几乎所有的方法都是静态方法,也就是说可以通过类名来调用相应的方法。
首先来看以下 copyOf 源码
public static <T> T[] copyOf(T[] original, int newLength) {
return (T[]) copyOf(original, newLength, original.getClass());
}
public static <T,U> T[] copyOf(U[] original, int newLength, Class<? extends T[]> newType) {
@SuppressWarnings("unchecked")
T[] copy = ((Object)newType == (Object)Object[].class)
? (T[]) new Object[newLength]
: (T[]) Array.newInstance(newType.getComponentType(), newLength);
System.arraycopy(original, 0, copy, 0,
Math.min(original.length, newLength));
return copy;
}
【参数解析】
original:原数组
newLength:数组长度
newType:数组类型
/----------------------------------------------------------/
可以看到 Arrays 中底层的 copyOf 方法首先保证新数组的类型和原数组是相等的,再调用 System.arraycopy 方法去复制数组。
Arrays 中都调用的 System 中的方法,那么两者有什么区别呢?
我们再来看一下 System.arraycopy 方法源码
public static native void arraycopy(Object src, int srcPos,
Object dest, int destPos,
int length);
【参数解析】
src:原数组
srcPos:开始复制的位置
dest:目标数组
destPos:复制的开始位置
length:数组长度
src和dest必须是同类型或者可以进行转换类型的数组
/----------------------------------------------------------/
我们可以看出 System.arraycopy 方法为本地方法,底层的原理现在学Java的我可能需要C同学的帮助啦🤭
使用总结
copyOf 方法 | arraycopy 方法 |
---|---|
调用时只需要原数组即可 | 调用时需要原数组和目标数组 |
有返回值 | 无返回值 |
新建一个对象,并调用arraycopy方法 | 底层方法 |
扩展
一维数组的复制
一维数组的四种复制方式:
- for循环挨个复制
- 调用 System.arraycopy 方法
- 调用 Arrays.copyOf 方法
- 调用 object 中的 clone 方法
for循环太简单啦,就没测试啦,同时我们可以看出一维数组的三种方式都是 深拷贝!
public static void main(String[] args){
int[] arr = {1,2,3};
int[] brr =new int[3];
System.arraycopy(arr,0,brr,0,arr.length);
int[] crr = Arrays.copyOf(arr, arr.length);
int[] drr = arr.clone();
arr[0] = 0;
System.out.println("arr: " +Arrays.toString(arr));
System.out.println("System.arraycopy: " + Arrays.toString(brr));
System.out.println("Arrays.copyOf: " + Arrays.toString(crr));
System.out.println("clone: " + Arrays.toString(drr));
}
运行结果
arr: [0, 2, 3]
System.arraycopy: [1, 2, 3]
Arrays.copyOf: [1, 2, 3]
clone: [1, 2, 3]