java类型擦除

咱们都知道无界通配符(unbounded wildcard) Set<?>能够处理任意类型数据,原生态类型(raw type)Set也能处理任意类型数据,可是他们区别是什么呢。

一、关于Set<?>的两个要点。

  • 由于符号?能够匹配任意类型,所以Set<?>能够处理任意类型的数据。
  • 由于咱们不知道?指的是什么类型,所以咱们不能忘Set里面添加任何类型的数据。

这两点看起来是否是很矛盾,其实并不矛盾,看下面的两个例子java

第一要点代码以下安全

//代码不报错
public static void main(String[] args) {
    HashSet<Integer> s1 = new HashSet<Integer>(Arrays.asList(1, 2, 3));
    printSet(s1);
 
    HashSet<String> s2 = new HashSet<String>(Arrays.asList("a", "b", "c"));
    printSet(s2);
}
 
public static void printSet(Set<?> s) {
    for (Object o : s) {
        System.out.println(o);
    }
}

第二要点代码以下code

public static void printSet(Set<?> s) {
    s.add(10);// 本行代码报错
    for (Object o : s) {
        System.out.println(o);
    }
}

这是由于咱们不知道?所指类型,所以除了null外的全部类型咱们都不能往里面添加。一样的,咱们不能初始化一个Set类型为?,以下blog

Set<?> set = new HashSet<?>();

二、Set vs Set<?>

看以下代码,是不会报错的。get

public static void printSet(Set s) {
    s.add("2");
    for (Object o : s) {
        System.out.println(o);
    }
}

由于原生态类型没有限制,因此它很容易是集合混乱,换句话说,无界通配符是安全的而原生态类型不是安全的。博客

三、什么时候使用Set<?>

当你使用泛型(generic type),可是你又不想关心实际泛型是什么的时候,使用无界通配符。以下:io

public static void main(String[] args) {
    HashSet<Integer> s1 = new HashSet<Integer>(Arrays.asList(1,2,3));
    HashSet<Integer> s2 = new HashSet<Integer>(Arrays.asList(4,2,3));
 
    System.out.println(getUnion(s1, s2));
}
 
public static int getUnion(Set<?> s1, Set<?> s2){
    int count = s1.size();
    for(Object o : s2){
        if(!s1.contains(o)){
            count++;
        }
    }
    return count;
转自个人博客园 http://cnblogs.com/qins/