一:asList 数组转ArrayList陷阱:
asList() 源码:
public static <T> List<T> asList(T... a) { return new ArrayList<T>(a); } private final E[] a; ArrayList(E[] array) { if (array==null) throw new NullPointerException(); a = array; }
分析:需要注意以下几点:
(1)总所周知,泛型对应的是对象类型,所以 asList(),转ArrayList的时候,数组的元素不能是基本数据类型。
(2) asList(),转换后的List不能进行add(),remove()等操作。返回一个受指定数组支持的固定大小的列表,对返回列表的更改会“直接写”到数组。
public static void test1(){ //1 asList(),转ArrayList的时候,数组的元素不能是基本数据类型。 int i[]={11,22,33}; List intList=Arrays.asList(i); System.out.println(intList.size());//结果:1,原因:asList将int i[] 当做了参数 } public static void test2(){ //2 asList(),转换后的List不能进行add(),remove()等操作。 //原因:返回一个受指定数组支持的固定大小的列表,对返回列表的更改会“直接写”到数组。 String s[]={"aa","bb","cc"}; List<String> sList=Arrays.asList(s); sList.add("dd");//error }
那么如何正确的将数组转ArraysList呢?
public static void test3(){ //方法一 String s[]={"aa","bb","cc"}; List<String> list1 = new ArrayList<String>(Arrays.asList(s)); list1.remove(0);//ok //方法二: List<String> list2 = new ArrayList<String>(s.length); list2.addAll(Arrays.asList(s)); list2.remove(0);//ok //方法三: List<String> list3 = new ArrayList<String>(s.length); Collections.addAll(list3, s); System.out.println(list3.remove(0));//ok //对于基本类型,可以使用apache commons-lang工具包 int i[]={11,22,33}; //先把基本类型转换一下,再用上面3种方法转换 Integer[] ii = ArrayUtils.toObject(i); List<Integer> list4 = new ArrayList<Integer>(ii.length); Collections.addAll(list4, ii);//ok }
commons-lang3-3.1.jar 下还有许多类似的工具。如:
二 list.toArray() 陷阱:
//java中的强制类型转换只能转换单个对象,所以不能使用这样的代码将 toArray 返回的数组强转为 String[] String[] arr=(String[])list.toArray();//会出现java.lang.ClassCastException
正确的用法:
public static void test4(){ String s[]={"aa","bb","cc"}; List<String> list1 = new ArrayList<String>(Arrays.asList(s)); String[] s1 = new String[list1.size()]; list1.toArray(s1); System.out.println(Arrays.toString(s1)); }
三 list.subList() 大陷阱:
注意:
(1)用此方法生成列表后,不要再去操作源列表(原因:可分析源码)
(2)用此方法生成列表list1,对list1进行的add,remove,最终还是在操作源列表。
看下面错误案例:
清单1:
public static void test6(){ String s[]={"aa","bb","cc","dd","ee"}; List<String> list1 = new ArrayList<String>(Arrays.asList(s)); List<String> list2 = list1.subList(0, 1); System.out.println("list2 size: "+list2.size()); list1.add("ff"); System.out.println("list2 size: " + list2.size());//error }
结果: java.util.ConcurrentModificationException
清单2:
public static void test5(){ String s[]={"aa","bb","cc","dd","ee"}; List<String> list1 = new ArrayList<String>(Arrays.asList(s)); List<String> list2 = list1.subList(0, 1); System.out.println("list1 size: "+list1.size()); System.out.println("list2 size: "+list2.size()); //用此方法生成列表list1,对list1进行的add,remove,最终还是在操作源列表。 list2.add("ff"); System.out.println("----操作list2-----"); System.out.println("list1 size: "+list1.size()); System.out.println("list2 size: "+list2.size()); }
结果:
list1 size: 5
list2 size: 1
----操作list2-----
list1 size: 6
list2 size: 2 需要操作子列表,需要拷贝一份出来:
清单3:
public static void test7(){ String s[]={"aa","bb","cc","dd","ee"}; List<String> list1 = new ArrayList<String>(Arrays.asList(s)); List<String> list2 = new ArrayList<String>(Arrays.asList(s)); //拷贝一份出来 list2.addAll(list1.subList(0, 1)); System.out.println("list1 size: "+list1.size()); System.out.println("list2 size: "+list2.size()); list2.add("ff"); System.out.println("----操作list2-----"); System.out.println("list1 size: "+list1.size()); System.out.println("list2 size: "+list2.size()); }
结果:
list1 size: 5
list2 size: 6
----操作list2-----
list1 size: 5
list2 size: 7