今天偶然发现ArrayList有两个toArray重载的方法。那么这两个方法有什么区别呢?
1 效率不同
再看看toArray()方法
public Object[] toArray() {
return Arrays.copyOf(elementData, size);
}
看第二行调用的copyOf()方法
public static <T> T[] copyOf(T[] original, int newLength) {
return (T[]) copyOf(original, newLength, original.getClass());
}
再看看toArray(T[] a)方法
public <T> T[] toArray(T[] a) {
if (a.length < size)
// Make a new array of a's runtime type, but my contents:
return (T[]) Arrays.copyOf(elementData, size, a.getClass());
System.arraycopy(elementData, 0, a, 0, size);
if (a.length > size)
a[size] = null;
return a;
}
看第四行调用的copyOf方法
public static <T,U> T[] copyOf(U[] original, int newLength, Class<? extends T[]> newType) {
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;
}
由代码可知,虽然toArray()方法最终也是调用的copyOf(U[] original, int newLength, Class< ? extends T[]> newType) 这个方法。但是这个方法中有个判断
T[] copy = ((Object)newType == (Object)Object[].class)
? (T[]) new Object[newLength]
: (T[]) Array.newInstance(newType.getComponentType(), newLength);
也就是说如果是已知类型,那么就会直接创建数组,而不知道类型就会根据反射创建数组。但是具体这个条件什么时候不成立我也不清楚
((Object)newType == (Object)Object[].class)
2 toArray(T[] a)会把list的值赋给数组a
1 虽然a会被赋值。但是只有当a.length>=list.size()是才会赋值给a.
System.arraycopy(elementData, 0, a, 0, size);
if (a.length > size)
a[size] = null;
return a;
2 如果a.length< list.size()那么数组a不会被赋值。
if (a.length < size)
// Make a new array of a's runtime type, but my contents:
return (T[]) Arrays.copyOf(elementData, size, a.getClass());
因为这里把数组拷贝一份就直接返回了。