先看看如下代码:在给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);
