Java泛型详解(下)

九. 泛型类型的继承规则

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

现在问题来了:
Pair<Manager>是Pair<Employee>的子类吗?

答案是:不是

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

Manager[] topHonchos = ...;
Pair<Employee> result = ArrayAlg.minmax(topHonchos); //Error
//minmax方法返回Pair<Manager>, 而不是Pair<Employee>, 这样的赋值是不合法的

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

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

十. 通配符类型

1. 通配符的概念:

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

public static void printBuddies(Pair<Employee> p) {
    Employee first = p.getFirst();
    Employee second = p.getSecond();
    System.out.println(first.getName() + "and" + second.getName() + "are buddies.");
}

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

public static void printBuddies(Pair<? extends Employee> p)

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

Pair<Manager> 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. 通配符捕获


参考文献:[美]Cay S. Horsimann著 《Java 核心技术 卷I》(第十版)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值