先看看如下代码:在给 list2 初始化大小后,再复制集合会抛出异常。
public static void main(String[] args) {
List<Integer> list = new ArrayList<>();
list.add(1);
list.add(2);
System.out.println(list);
List<Integer> list2 = new ArrayList<>(list.size());
Collections.copy(list2,list);
System.out.println(list2);
}
[1, 2]
Exception in thread "main" java.lang.IndexOutOfBoundsException: Source does not fit in dest
at java.util.Collections.copy(Collections.java:558)
at com.jiaozi.cashloan.worker.risk.feature.application.ProxyFeatures.main(ProxyFeatures.java:70)
可以看到 ArrayList 构造方法初始化集合大小,设置的的大小是内部数组(Object [] elementData)的大小,并不是集合 size 的大小,此时 size==0;
transient Object[] elementData; // non-private to simplify nested class access
public ArrayList(int initialCapacity) {
if (initialCapacity > 0) {
this.elementData = new Object[initialCapacity];
} else if (initialCapacity == 0) {
this.elementData = EMPTY_ELEMENTDATA;
} else {
throw new IllegalArgumentException("Illegal Capacity: " + initialCapacity);
}
}
再去调用 copy 方法时,此时 ArrayList 的 size==0,因此抛出异常。
public static <T> void copy(List<? super T> dest, List<? extends T> src) {
int srcSize = src.size();
if (srcSize > dest.size())
throw new IndexOutOfBoundsException("Source does not fit in dest");
····
}
可以看到 2 > 0 ,抛出 IndexOutOfBoundsException 异常。
注意事项:
- ArrayList 声明时默认的元素个数是 0,只有进行 add(),remove()操作时 size()的大小才有相应的变化。
- 进行集合复制时,一要声明目的集合的元素的个数,并且要等于或者大于源集合的元素的个数。
- 如果不声明或者小于源集合的元素个数,这样就会报错,报下标界的异常(java.lang.IndexOutOfBoundsException)。
正确使用 Collections.copy (),的二种方式:
- 第一种
ArrayList<String> arrayList2 = new ArrayList<>(Arrays.asList(new String[arrayList1.size()]));
Collections.copy(arrayList2, arrayList1);
- 第二种
ArrayList<String> arrayList2 = new ArrayList<>();
Collections.addAll(arrayList2,new String[arrayList1.size()]);
Collections.copy(arrayList2, arrayList1);