Java override方法时返回类型协变、实参逆变

Covariant(协变) 与 Contravariant(逆变)

逆变与协变用来描述类型转换(type transformation)后的继承关系:(从继承关系来看,继承树从上到下,从父类到子类是协变,从子类到父类则是逆变)

  • 当本使用父类的地方,却使用了子类,则称为Covariant(协变)。
  • 当本实用子类的地方,却使用了父类,则称为Contravariant(逆变)。
  • 当使用类没有发生变化,则称为invariant(不变)。

具体到Java语言中,则是:

  • 当子类覆盖父类方法时,返回类型可以协变(假设A是B的父类,父类方法返回A,子类覆盖父类方法时却返回B,从返回类型A到返回类型B,发生了协变;当然也可以不变,这是肯定得)。
  • 当实参传递给形参,参数类型可以逆变(实参类型为子类,而形参类型为父类,从子类到父类,发生了逆变;注意这里与继承覆盖这些无关)。

以上两点也符合协变与逆变的本质——里氏替换原则:所有引用基类的地方必须能透明地使用其子类的对象。即Java中父类引用可以指向子类对象。

协变例子(override方法时返回类型协变)

class Grain {
  public String toString() { return "Grain"; }
}

class Wheat extends Grain {
  public String toString() { return "Wheat"; }
}

class Mill {
  Grain process() { return new Grain(); }
}

class WheatMill extends Mill {
  @override
  Wheat process() { return new Wheat(); }
  //返回值Wheat是父类方法返回值Grain的子类,发生协变
}

public class CovariantReturn {
  public static void main(String[] args) {
    Mill m = new Mill();
    Grain g = m.process();
    System.out.println(g);
    m = new WheatMill();
    g = m.process();
    System.out.println(g);
  }
} /* Output:
Grain
Wheat
*/

注意WheatMill 覆盖子类方法时,返回值类型发生了协变。

逆变例子(实参逆变)

class Grain {
    public String toString() { return "Grain"; }
}

class Wheat extends Grain {
    public String toString() { return "Wheat"; }
}

class Mill {
    void process(Grain g) { System.out.println("Mill "+g); }
}

public class Ex10 {
    public static void main(String[] args) {
        Wheat w = new Wheat();
        new Mill().process(w);
    }
} /* Output:
Mill Wheat
*/

new Mill().process(w)实参类型为子类,而形参类型为父类,从子类Wheat到父类Grain,发生了逆变。

逆变不能用在覆盖父类方法的参数类型

class Grain {
    public String toString() { return "Grain"; }
}

class Wheat extends Grain {
    public String toString() { return "Wheat"; }
}

class Mill {
    void process(Wheat w) { System.out.println("Mill "+w); }
}

class WheatMill extends Mill {
    //@Override 注意加此句报错,因为父类没有这样的签名的方法
    void process(Grain g) { System.out.println("WheatMill "+g); }//此方法从未被使用,应该解释为函数重载
}

public class Ex10 {
    public static void main(String[] args) {
        Wheat w = new Wheat();
        new Mill().process(w);
        new WheatMill().process(w);
    }
}/* Output:
Mill Wheat
Mill Wheat
*/

override方法时,父类类型参数和子类类型参数是严格区分开来了的。所以,这个例子实际展示了overload重载函数。

<think>嗯,用户问的是Java中的返回类型,我需要先回忆一下这个概念的定义。返回类型可能和方法的覆盖有关,对吧?我记得在Java早期版本中,子类覆盖父类方法返回类型必须完全一致。但是后来某个版本开始,允许子类方法返回类型是父类方法返回类型的子类,这样更灵活。 用户可能是Java开发者,遇到需要覆盖方法但希望返回更具体类型的情况,比如父类方法返回Animal,子类返回Dog。这返回类型就派上用场了。需要确认Java哪个版本引入的这个特性,应该是Java 5之后,因为之前是不支持的。 接下来要解释返回类型的具体例子,说明如何工作。比如父类有一个方法返回Animal,子类覆盖返回Dog,而Dog是Animal的子类。这样在调用子类方法,可以直接得到Dog类型,不需要强制转换,提高了代码的简洁性和安全性。 还要注意和重载的区别,返回类型方法覆盖,方法签名除了返回类型不同,其他部分必须相同。而重载是方法名相同但参数不同。这里要澄清可能存在的混淆点。 用户可能想知道为什么返回类型有用,比如在工厂方法模式中,子类可以返回具体产品类型,而不是父类的抽象类型,这样客户端代码可以直接使用子类特有的方法,而无需向下转型,减少出错可能。 另外,可能需要提到编译器的处理方式,确保在运行正确解析返回类型,以及返回类型在面向对象设计中的优势,比如更清晰的继承结构和类型安全。 需要检查是否有其他相关概念需要对比,比如逆变参数类型,但Java不支持方法的参数类型逆变,所以可以不用深入,除非用户问起。 最后总结返回类型的定义、作用、使用场景和好处,确保用户能清晰理解并应用到实际编码中。</think>在 Java 中,**返回类型(Covariant Return Type)** 是一种允许子类**重写父类方法返回更具体类型**的特性。这是对传统方法重写规则的扩展,旨在提高代码的灵活性和类型安全性。 --- ### 核心概念 - **传统规则**: 子类重写父类方法,必须保持**返回类型完全一致**(Java 5 之前的限制)。 - **返回类型**(Java 5+): 子类重写父类方法,可以返回父类方法返回类型的**子类**(更具体的类型)。 --- ### 示例代码 ```java class Animal { public Animal reproduce() { return new Animal(); } } class Dog extends Animal { // 返回类型:返回 Dog(Animal 的子类) @Override public Dog reproduce() { return new Dog(); } } ``` --- ### 关键特点 1. **类型细化**: 子类方法返回的类型比父类更具体(如 `Dog` 是 `Animal` 的子类)。 2. **类型安全**: 调用子类方法,可以直接使用具体类型,无需强制类型转换: ```java Dog dog = new Dog(); Dog puppy = dog.reproduce(); // 直接返回 Dog 类型 ``` 3. **兼容性**: 返回类型不违反里氏替换原则(LSP),因为子类返回的类型始终满足父类约束。 --- ### 典型应用场景 - **工厂方法模式**: 父类定义抽象工厂方法返回通用类型,子类返回具体类型: ```java abstract class ShapeFactory { public abstract Shape create(); } class CircleFactory extends ShapeFactory { @Override public Circle create() { // 返回具体类型 Circle return new Circle(); } } ``` - **链式调用**: 支持子类方法返回自身类型以实现链式调用: ```java class Builder { Builder setWidth(int width) { /* ... */ return this; } } class AdvancedBuilder extends Builder { @Override AdvancedBuilder setWidth(int width) { // 返回 AdvancedBuilder super.setWidth(width); return this; } } ``` --- ### 对比传统方法重写 | **场景** | **传统重写** | **返回类型** | |-----------------------|-----------------------|-----------------------| | 返回类型 | 必须完全一致 | 允许子类细化类型 | | 类型转换 | 需要强制转换 | 直接使用具体类型 | | 代码灵活性 | 较低 | 更高 | --- ### 总结 返回类型通过允许子类方法返回更具体的类型,实现了: - **类型安全**:避免不必要的强制类型转换。 - **代码简洁性**:直接操作具体类型。 - **设计灵活性**:支持更优雅的继承和多态实现。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值