1.关于hashCode和equals的处理,请遵循以下规则
1.1只要重写equals,就必须重写hashCode
1.2 因为Set存储的是不重复的对象,依据hashCode和equals进行判断,所以Set存储的对象必须重写这两个方法
1.3 如果自定义对象作为Map的键,那么必须重写hashCode和equals
说明:String重写了hashCode和equals方法,所以我么可以非常愉快的使用String对象作为key来使用;
2.ArrayList的subList结果不可强转成ArrayList,否则会抛出ClassCastException异常,即java.util.RandomAccessSubList cannot be cast to java.util.ArrayList。
说明:subList返回的是ArrayList的内部类SubList,并不是ArrayList而是ArrayList的一个视图,对于SubList子列表的所有操作最终会反映到原列表上;
3.在SubList场景中,高度注意对原集合元素的增加或删除,均会导致子列表的遍历、增加、删除产生ConcurrentModificationException异常。
4.使用集合转数组的方法,必须使用集合的toArray(T[ ] array),传入的是类型完全一样的数组,大小就是list.size()。
说明:使用toArray带参方法,入参分配的数组空间不够大的时候,toArray方法内部将重新分配内存空间,并返回新数组地址,如果数组元素个数大于实际所需,下标为[ list.size() ]的数组元素将被置为null,其他数组元素保持原值,因此最好将方法入参数组大小定义与集合元素个数一致;
正确写法示范:
List<String> list = new ArrayList<String>(2);
list.add("guan");
list.add("bao");
String[] array = new String[list.size()];
array = list.toArray(array);
反例:直接使用toArray无参方法存在问题,此方法返回值只能是Object[]类,若强转其他类型数组将出现ClassCastException错误;
5.使用工具类Arrays.asList()把数组转换为集合的时候,不能使用其修改集合相关的方法,它的add/remove/clear方法会抛出UnsupportedOperationException异常
说明:asList的返回对象是一个Arrays内部类,并没有实现集合的修改方法.Arrays.asList体现的是适配器模式,只是转换接口,后台的数据仍是数组。
案例:
String [] str = new String []{"123","456"};
List<String> list= Arrays.asList(str);
在这串代码的基础上
执行list.add("strine"):就会出现运行时异常
如果执行str[0]="strine"; 那么list.get(0)的时候会发现,也随之修改了;
6.泛型通配符<? extends T>来接收返回的数据,此写法的泛型集合不能使用add方法,而<? super T>不能使用get方法,作为接口调用赋值时容易出错
说明:扩展说一下PECS(Producer Extends Consumer Super)原则:第一、频繁往外读取内容的,适合用<? extends T>,第二、经常往里插入的,适合用<? super T>;
7.不要再foreach循环里面进行元素的remove/add操作。remove元素请使用Iterator操作,并且如果 并发操作,还需要对Iterator对象加锁
8.在JDK7版本以上,Comparator实现类要满足以下三个条件,否则Arrays.sort,Collections.sort会抛IllgalArgumentException异常:
x,y的比较结果和y,x的比较结果相反;
x>y,y>Z,则x>Z;
x=y,则x,z比较结果和y,z比较结果相同;
9.集合泛型定义的时候,在JDK7及以上,使用diamond语法或全省略;
说明:菱形泛型,及diamond,直接使用<>来指代前边已经指定的类型。
案例:
10.集合初始化的时候,指定集合初始值的大小
说明:HashMap使用HashMap(int initialCapacity)初始化;
案例:initialCapacity=(需要存储的元素个数/负载因子)+1,注意:负载因子默认为0.75,如果暂时无法确定初始值大小,请设置为16(即默认值);
反例:HashMap需要放置1024个元素,由于没有设置容量初始值大小,随着元素不断增加,容量7次被迫扩大,resize需要重建hash表,严重影响性能。
11.使用entrySet遍历Map类集合KV,而不是keySet方式进行遍历。
说明:keySet其实是遍历了两次,一次是转为Iterator对象,另一次是从hashMap中取出key所对应的value。而entrySet只是遍历了一次就把key和value都放到了entry中,效率更高,如果是JDK8,就 使用Map.foreach方法;
案例:values()返回的是V值集合,是一个list集合对象,keySet()返回的是k值集合,是一个Set集合对象,entrySet()返回的是K-V值组合集合。
12.高度注意Map类集合K/V能不能存储null值的情况,如下:
尤其要注意HashMap和ConcurrentHashMap!
13.合理利用好集合的有序性和稳定性,避免集合的无序性和不稳定性带来的负面影响;
说明:有序性是指遍历的结果是按某种比较规则依次排列的,稳定性值集合每次遍历的元素次序是一定的。