Java 8 - 两个接口包含具有相同方法签名但返回类型不同的默认方法,如何覆盖?

时间:2017-12-17 14:42:14

标签: java interface java-8 override default-method

我理解如果一个类实现了包含相同名称的默认方法的多个接口,那么我们需要在子类中重写该方法,以便明确定义我的方法将做什么。
问题是,请参阅下面的代码:

interface A {
    default void print() {
        System.out.println(" In interface A ");
    }
}

interface B {
    default String print() {
        return "In interface B";
    }
}

public class C implements A, B {

    @Override
    public String print() {
        return "In class C";
    }

    public static void main(String arg[]) {
        // Other funny things
    }
}

现在接口A和B都有一个名为'print'的默认方法,但我想覆盖接口B的print方法 - 返回字符串并按原样保留A的打印方式。但是这段代码没有编译给出这个:

Overrides A.print
The return type is incompatible with A.print()

很明显,编译器试图覆盖A的打印方法,我不明白为什么!

1 个答案:

答案 0 :(得分:3)

这是不可能的。

8.4.8.3

  

如果返回类型为d1的方法声明R1覆盖或隐藏另一个返回类型为d2的方法R2的声明,则d1必须为返回类型可替换 d2,或发生编译时错误。

8.4.5

  

返回类型为d1的方法声明R1为其他方法d2 return-type-substitutable ,返回类型为R2 iff any以下是真实的:

     
      
  • 如果R1void,则R2void

  •   
  • 如果R1是基本类型,则R2R1相同。

  •   
  • 如果R1是引用类型,则以下之一为真:

         
        
    • R1,适用于d2的类型参数,是R2的子类型。

    •   
    • R1可以通过未选中的转化转换为R2的子类型。

    •   
    • d1d2R1 = |R2|的签名不同。

    •   
  •   

换句话说,void,原始和引用返回方法只能覆盖并被相同类别的方法覆盖。 void方法可能只覆盖另一个void方法,引用返回方法可能只覆盖另一个引用返回方法,依此类推。

您遇到问题的一种可能解决方案可能是使用合成而不是继承:

class C {
    private A a = ...;
    private B b = ...;
    public A getA() { return a; }
    public B getB() { return b; }
}