集合一些方法陷阱

Posted xiaozhuanfeng

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了集合一些方法陷阱相关的知识,希望对你有一定的参考价值。

一: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  
   
 
 

以上是关于集合一些方法陷阱的主要内容,如果未能解决你的问题,请参考以下文章

Java集合与泛型中的几个陷阱,你掉进了几个?

[ jquery 文档处理 insertBefore(content) before(content|fn) ] 此方法用于把所有匹配的元素插入到另一个指定的元素元素集合的前面,实现外部插入(代码片段

带有红宝石集合/可枚举的酷技巧和富有表现力的片段[关闭]

七个Swift中的陷阱以及避免方法

代码片段 - Golang 实现集合操作

java集合示例 小心重载的陷阱