Arrays.copyOf()方法详解-jdk1.8

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;
}

可以看到,最终调用的是System.arraycopy()方法,关于这个方法使用,可查看System.arraycopy详解

1. 方法的含义
返回一个类型为T的数组,数组的容量为newLength和original.length中的小值,元素为original数组中的元素(取0到newLength和original.length中的小值)

说白了,这个方法就是为了数组元素的向上转型,还有就是截断数组

2. (Object)newType == (Object)Object[].class
判断newType是否是Object数组

3. 为什么要加(Object)呢?
因为要用使用==去比较它们的内存地址,从而判断它们是不是同一类型,而使用==,就要向上强转为Object,不然编辑器不让通过,不能比较

public class SystemArrayCopy {

    public static void main(String[] args) {
        System.out.println(Object.class == Integer.class);
    }
}
/*
编辑时报错:
Error:(37, 41) java: 不可比较的类型: java.lang.Class<java.lang.Object>和java.lang.Class<java.lang.Integer>
*/

加上(Object)后

public class SystemArrayCopy {

    public static void main(String[] args) {
        System.out.println(Object.class == (Object)Integer.class);
    }
}
/*
false
*/

4. newType.getComponentType()

public native Class<?> getComponentType();

本地方法,返回数组内的元素类型,不是数组时,返回null

public class SystemArrayCopy {

    public static void main(String[] args) {
        String[] o = {"aa", "bb", "cc"};
        System.out.println(o.getClass().getComponentType());
        System.out.println(Object.class.getComponentType());
    }
}
/*
class java.lang.String
null
*/

5. Array.newInstance(newType.getComponentType(), newLength)
创建一个类型与newType一样,长度为newLength的数组

public static Object newInstance(Class<?> componentType, int length)
    throws NegativeArraySizeException {
    return newArray(componentType, length);
}
private static native Object newArray(Class<?> componentType, int length)
        throws NegativeArraySizeException;

Array.newInstance内部直接调用Array.newArray,newArray为本地方法,由虚拟机实现

newInstance返回为Object,实质为数组

public class SystemArrayCopy {

    public static void main(String[] args) {
        Object o = Array.newInstance(String.class, 10);
        System.out.println(o.getClass());
        System.out.println(String.class); // 用于对比
    }
}
/*
class [Ljava.lang.String;
class java.lang.String
*/

可以看到,Array.newInstance的返回虽然是Object类型,但是它实质上是String数组,可以强制转换成String[],如:String[] arr = (String[]) Array.newInstance(String.class, 10);

6. (T[]) new Object[newLength],为什么Object[]可以强制转换成T[]呢?
判断(Object)newType == (Object)Object[].class
为true时,执行的是(T[]) new Object[newLength],那T就是Object,newType也是Object[].class,所以可以强转成T[]
为false时,执行的是(T[]) Array.newInstance(newType.getComponentType(), newLength),Array.newInstance返回的本质就是T[],所以可以强转成T[]
7. 为什么强转为T[],而不是Object[]?
因为我这个方法就是要返回T[],这是方法编写的本意

8. 作用
把源数组中元素的类型向上转型
截断数组,当给定长度小于给定数组时,就可以实现截断的效果

如:ArrayList底层数组的类型转换使用的这个方法(ArrayList参数为集合的构造函数中)

转载于:https://my.oschina.net/u/2935389/blog/3030649

  • 3
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Arrays.copyOf() 和 System.arraycopy() 都是用于数组的复制操作,但它们有一些不同之处。 Arrays.copyOf() 方法是在 Java 1.6 版本中引入的,它用于创建一个新的数组,并将源数组中的元素复制到新数组中。该方法具有以下两种重载形式: 1. copyOf(original, newLength):将原始数组的前 newLength 个元素复制到新数组中。如果新长度小于原始数组的长度,则新数组将被截断;如果新长度大于原始数组的长度,则新数组将被填充默认值。 2. copyOf(original, newLength, newType):与上述方法相似,但可以指定新数组的类型。 示例使用 Arrays.copyOf() 方法: ```java int[] original = {1, 2, 3, 4, 5}; int[] copy1 = Arrays.copyOf(original, 3); // 复制原数组的前三个元素 int[] copy2 = Arrays.copyOf(original, 7); // 复制原数组的所有元素,并用默认值填充额外位置 ``` System.arraycopy() 方法是在 Java 1.0 版本中引入的,它也用于将源数组中的元素复制到目标数组中。该方法的语法如下: ```java System.arraycopy(src, srcPos, dest, destPos, length); ``` 其中,src 是源数组,srcPos 是源数组中要开始复制的起始位置,dest 是目标数组,destPos 是目标数组中要开始粘贴的起始位置,length 是要复制的元素个数。 示例使用 System.arraycopy() 方法: ```java int[] src = {1, 2, 3, 4, 5}; int[] dest = new int[5]; System.arraycopy(src, 0, dest, 0, 5); // 复制 src 数组的所有元素到 dest 数组中 ``` 总结来说,Arrays.copyOf() 方法提供了更简洁的方式来复制数组,并且可以轻松地截取或填充数组。而 System.arraycopy() 方法则提供了更灵活的复制方式,可以指定复制的起始位置和复制的元素数量。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值