2

泛型、继承和子类型

如你所知,只要类型兼容,就可以将一种类型的对象分配给另一种类型的对象,例如,你可以将Integer分配给Object,因为ObjectInteger的超类型之一:

Object someObject = new Object();
Integer someInteger = new Integer(10);
someObject = someInteger;   // OK

在面向对象的术语中,这被称为“是一种”关系,由于Integer是一种Object,因此允许赋值,但是Integer也是一种Number,所以下面的代码也是有效的:

public void someMethod(Number n) { /* ... */ }

someMethod(new Integer(10));   // OK
someMethod(new Double(10.1));   // OK

泛型也是如此,您可以执行泛型类型调用,将Number作为其类型参数传递,如果参数与Number兼容,则允许任何后续的add调用:

Box<Number> box = new Box<Number>();
box.add(new Integer(10));   // OK
box.add(new Double(10.1));  // OK

现在考虑以下方法:

public void boxTest(Box<Number> n) { /* ... */ }

它接受什么类型的参数?通过查看其签名,你可以看到它接受一个类型为Box<Number>的参数,但是,这是什么意思?你是否可以按照预期传递Box<Integer>Box<Double>?答案是“否”,因为Box<Integer>Box<Double>不是Box<Number>的子类型。

在使用泛型编程时,这是一个常见的误解,但这是一个重要的学习概念。

generics-subtypeRelationship.gif

给定两个具体类型AB(例如,NumberInteger),MyClass<A>MyClass<B>无关,无论AB是否相关,MyClass<A>MyClass<B>的公共父级是Object

有关如何在类型参数相关时在两个泛型类之间创建类似子类型关系的信息,请参阅通配符和子类型。

泛型类和子类型

你可以通过扩展或实现泛型类或接口来对其进行子类型化,一个类或接口的类型参数与另一个类或接口的类型参数之间的关系由extendsimplements子句确定。

使用Collections类作为示例,ArrayList<E> implements List<E>List<E> extends Collection<E>,因此ArrayList<String>List<String>的子类型,也是Collection<String>的子类型,只要不改变类型参数,就会在类型之间保留子类型关系。

generics-sampleHierarchy.gif

现在假设我们想要定义我们自己的列表接口PayloadList,它将泛型类型P的可选值与每个元素相关联,它的声明可能如下:

interface PayloadList<E,P> extends List<E> {
  void setPayload(int index, P val);
  ...
}

PayloadList的以下参数化是List<String>的子类型:

  • PayloadList<String,String>
  • PayloadList<String,Integer>
  • PayloadList<String,Exception>

generics-payloadListHierarchy.gif


上一篇:有界类型参数
下一篇:类型推断

博弈
2.5k 声望1.5k 粉丝

态度决定一切