java泛型详解_Java泛型详解(下)

九. 泛型类型的继承规则

假设现在有一个类Employee和它的子类Manager

现在问题来了: Pair是Pair的子类吗?

答案是:不是

例如,下面的代码将不会编译成功:

Manager[] topHonchos = ...;

Pair result = ArrayAlg.minmax(topHonchos); //Error

//minmax方法返回Pair, 而不是Pair, 这样的赋值是不合法的

也就是说,无论 S 与 T 有什么联系,通常,Pair与Pair没有什么联系 也可以这么理解,无论赋给类型变量的类型之间有什么联系,在泛型类里,这些关系都不存在了

另外,泛型类可以扩展或实现其他的泛型类,例如:ArrayList类实现了List接口,这意味着,一个ArrayList可以被转换为一个List。但是,一个ArrayList不是一个ArrayList或List

十. 通配符类型

1. 通配符的概念:

假设我们现在需要打印一对员工的方法:

public static void printBuddies(Pair p) {

Employee first = p.getFirst();

Employee second = p.getSecond();

System.out.println(first.getName() + "and" + second.getName() + "are buddies.");

}

现在问题来了,Manager虽然不同于普通员工,但他也算是员工,他是员工的子类,但是根据之前说的泛型类继承规则,Pair与Pair没有任何关系,所以这个方法就不能接受Pair类了,难道Manager与普通员工就不能做朋友了吗?不,为了解决这个问题,我们使用通配符类型:

public static void printBuddies(Pair extends Employee> p)

类型Pair是Pair的子类型

可以认为,通配符为泛型类之间添加了一层联系,然而这种联系并不纯粹,我们来看一个例子:

Pair managerBuddies = new Pair<>(CEO, CFO); //JDK7之后可以省略构造函数的类型变量,由编译器根据语句自己翻译

Pair extends Employee> wildcBuddies; //OK

wildcardBuddies.setFirst(lowlyEmployee); //这步就会报错,compile-time error

究其原因,我们来看对于类型Pair extends Employee>内部对 setFirst 和 getFirst 方法的调用,它是这样的:

? extends Employee getFirst() //将getFirst返回值赋给一个Employee的引用完全合法

void setFirst(? extends Employee) //出现类型错误,因为编译器只知道需要某个Employee的子类型,但不知道具体是什么类型

2. 通配符的超类型限定

通配符还有一个比较强的功能就是可以指定一个超类,像这样:

? super Manager //super关键字限制通配符为Manager的所有超类(父类)

这样的做法与之前的子类型限定恰好相反。这样可以为方法提供参数(setFirst),但不能使用返回值(getFirst)

void setFirst(? super Manager)

? super Manager getFirst()

直观的讲,带有超类型限定的通配符可以向泛型对象写入,带有子类型限定的通配符可以从泛型对象读取

3. 无限定通配符

还可以使用无限定通配符,例如,Pair>,初看起来,这好像与原始的Pair类型一样,实际上却有很大的不同,类型Pair>有以下的方法:

? getFirst()

void setFirst(?)

getFisrt的返回值只能赋给一个Object。setFirst方法不能被调用,甚至不能用Object调用。Pair>和Pair的本质不同在于:可以用任意Object对象调用原始Pair类的setObject方法

4. 通配符捕获


版权声明:本文为weixin_33467739原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。