Java泛型详解

2.6 Java泛型详解

Java泛型是JDK5中引入的一个新特性,允许在定义类和接口的时候使用类型参数(type parameter),声明的类型参数在使用时用具体的类型来替换。泛型最主要的应用是在JDK5中的新集合类框架中。

2.6.1 类型擦除

首先我们看一下Java泛型中的类型擦除:在生成的Java字节码中是不包含泛型的类型信息的,使用泛型的时候加上的类型参数在编译的时候会被编译器去掉,这个过程就称为类型擦除。如在代码中定义的List和List等类型,在编译之后都会变成List。JVM看到的只是List,而由泛型附加的类型信息对JVM来说是不可见的(反射是可见的,具体可参见2.3中笔记),这一点和C++模板机制实现泛型有很大区别。

很多泛型的奇怪特性都与类型擦除有关,比如:

  1. 泛型类并没有自己独有的Class类对象。比如并不存在List.class或者List.class,而只有List.class。
  2. 静态变量是被泛型类的所有实例所共享的。对于声明为MyClass的类,访问其中的静态变量的方法仍然是MyClass.myStaticVar。不管是通过new MyClass还是new MyClass创建的对象,都是共享一个静态变量。
  3. 泛型的类型参数不能用在Java异常处理的catch语句中。因为异常处理是由JVM在运行时刻来进行的。由于类型信息被擦除,JVM无法区分两个异常类型MyException和 MyException的,对于JVM来说,它们都是MyException类型的,也就无法执行与异常对应的catch语句。

类型擦除的基本过程也比较简单,首先是找到用来替换类型参数的具体类,一般是Object,如果指定了类型参数的上界的话,则是这个上界。然后把代码中的类型参数都替换成该类。同时去掉出现的类型声明,即去掉<>的内容。比如T get()方法声明就变成了Object get();List就变成了List。接下来就可能需要生成一些桥接方法(bridge method),这是由于擦除了类型之后的类可能缺少某些必须的方法。比如考虑下面的代码:

class MyString implements Comparable {
    @Override
    public int compareTo(String str) {
        return 0;
    }
}

当类型信息被擦除之后,上述类的声明变成了class MyString implements Comparable。但是这样的话,类MyString就会有编译错误,因为没有实现接口Comparable声明的int compareTo(Object)方法,这个时候就由编译器来动态生成这个方法。

2.6.2 通配符与上下界

在使用泛型类的时候,既可以指定一个具体的类型,也可以用通配符?来表示未知类型,如List就声明了List中包含的元素类型是未知的。通配符所代表的其实是一组类型,但具体的类型是未知的。通配符分为三类:无界通配符、上界通配符和下界通配符。通配符本身比较复杂,我们会以集合为代表,以元素的添加和获取为例简要说明其用法。

无界通配符

“?”表示无界通配符,List表示:List中存储的元素的类型是未知的。

  1. 添加元素,使用无界通配符时,由于类型不确定的(可以是任何类型),不可以向List添加任何元素(除了null),因为如果它包含了String的话,往里面添加Integer显然是错误的。那为什么不能添加Object引用,是因为Object引用可以指向子类实例,编译期是无法获知其具体类型,所以为了类型安全,其不可以添加任何元素(除了null)。
  2. 获取元素,List中的元素只可以使用Object来引用,因为其元素肯定是Object及其子类引用。

事实上无界通配符通常会用在以下两种情况:

  1. 在业务逻辑与泛型类型无关,如List.size和List.clean等。实际上,最常用的就是Class,因为Class并没有依赖于T。
  2. 当方法参数是原始的Object类型,如下:
public static void printList(List list) {
    for (Object elem : list)
        System.out.println(elem + "");
}

//使用泛型类替换
public static void printList(List list) {
    for (Object elem: list)
        System.out.print(elem + "");
}
 
 

这样就可以兼容更多的输出,而不单纯是List,如下:

List li = Arrays.asList(1, 2, 3);
List  ls = Arrays.asList("one", "two", "three");
printList(li);
printList(ls);

上界通配符

“? extends Animal”表示通配符的上界是Animal,即“? extends Animal”可以代表Animal及其子类,不能代表Animal父类。
首先要阐明一点,上界通配符和下界通配符更多的是为了解决泛型不协变的问题。

  1. 添加元素,使用上界通配符时,其类型仍然是不确定的(会是某个类型及其子类型),所以仍然不可以向List<? extends Animal>添加任何元素(null除外),因为如果它包含了Cat的话,往里面添加Dog显然是错误的,同样不可以添加Object引用。
  2. 获取元素,List中的元素可以用Animal来引用的,因为其元素肯定是Animal及其子类引用。

而且引入了上界之后,在使用类型的时候就可以使用上界类中定义的方法,因为其中元素肯定是Animal类或其子类成员引用。

下界通配符

“? super Animal”表示通配符的下界是Animal,即“? super Animal”可以代表Animal及其父类,不能代表Animal子类。

  1. 添加元素,使用下界通配符时,虽然类型仍然是不确定的(会是某个类型及其父类),但是此时可以向List<? super Animal>添加Animal或者其子类型Dog元素,因为如果它包含了Animal的话,往里面添加Dog显然是可以的,子类型可以替换父类型。
  2. 获取元素,List中的元素只可以用Object来引用的,“? super Animal”可以代表Animal及其父类,所以只能通过Object来进行应用。

总结

PECS(Producer Extends Consumer Super)原则:频繁往外读取内容的,适合用上界Extends;经常往里插入的,适合用下界Super。

通配符是实参,通配符这些看起来很奇怪的特性原因在于:编译器要保证类型安全,Object类型是所有类型的祖宗类型,而父类引用是可以引用子类对象的。

2.6.3 类型系统

在Java中,大家比较熟悉的是通过继承机制而产生的类型体系结构,比如String继承自Object。根据Liskov替换原则,子类是可以替换父类的。当需要Object类的引用的时候,如果传入一个String对象是没有任何问题的。但是反过来的话,即用父类的引用替换子类引用的时候,就需要进行强制类型转换。这种自动的子类替换父类的类型转换机制,对于数组也是适用的(前面说过数组是协变的),String[]可以替换Object[]。

而泛型的引入,对于这个类型系统产生了一定的影响,正如前面提到的List是不能替换掉List的。引入泛型之后的类型系统增加了两个维度:一个是类型参数自身的继承体系结构,另外一个是泛型类或接口的继承体系结构。前者是指对于List和List这样的情况,类型参数String是继承自Object的。而后者是指List接口继承自Collection接口。对于这个类型系统,有如下的一些规则:

  1. 相同类型参数的泛型类的关系取决于泛型类自身的继承体系结构,即List是Collection的子类型,List可以替换Collection,这种情况也适用于带有上下界的类型声明。
  2. 当泛型类的类型声明中使用了通配符的时候,其子类型可以在两个维度上分别展开。如对Collection来说,其子类型可以在Collection这个维度上展开,即List和Set等;也可以在Number这个层次上展开,即Collection和Collection等。如此循环下去,ArrayList和HashSet等也都算是Collection的子类型。
  3. 如果泛型类中包含多个类型参数,则对于每个类型参数分别应用上面的规则。

2.6.4 开发自己的泛型类

泛型类与一般的Java类基本相同,只是在类和接口定义上多出来了用<>声明的类型参数。一个类可以有多个类型参数,如MyClass。每个类型参数在声明的时候可以指定上界。所声明的类型参数在Java类中可以像一般的类型一样作为方法的参数和返回值,或是作为域和局部变量的类型。但是由于类型擦除机制,类型参
数并不能用来创建对象或是作为静态变量的类型。考虑下面的泛型类中的正确和错误的用法。

class ClassTest {
    private X x; //正确用法
    private static Y y; //编译错误,不能用在静态变量中
    public X getFirst() { //正确用法
        return x;
    }
    public void wrong() {
        Z z = new Z(); //编译错误,不能创建对象
    }
}

2.6.5 最佳实践

在使用泛型的时候可以遵循一些基本的原则,从而避免一些常见的问题。

  1. 在代码中避免泛型类和原始类型的混用。比如List和List不应该共同使用,这样会产生一些编译器警告和潜在的运行时异常。
  2. 在使用带通配符的泛型类的时候,需要明确通配符所代表的一组类型的概念。由于具体的类型是未知的,很多操作是不允许的。
  3. 泛型类最好不要同数组一块使用。你只能创建new List[10]这样的数组,无法创建new List[10]这样的。这限制了数组的使用能力,而且会带来很多费解的问题。因此当需要类似数组的功能时候,使用集合类即可。

你可能感兴趣的:(Java泛型详解)