JDK8 类和接口的多继承

本文探讨了Java8中引入的接口多继承特性及其带来的挑战。通过实例展示了当子接口继承的父接口中有同名方法时如何解决冲突,并讨论了类继承与接口继承之间的优先级。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

JDK8 类和接口的多继承

以前经常谈论的 Java 对比 C++ 的一个优势是 Java 中没有多继承的问题. 因为 Java 中子类只能继承 (extends) 单个父类, 尽管可以实现 (implement) 多个接口, 但是接口之中只有抽象方法, 方法体是空的, 没有具体的方法实现, 不会有方法冲突的问题.

在 Java8 发布后, 接口中也可以定义方法了(default method). 之所以打破之前的设计在接口中增加具体的方法, 是为了既有的成千上万的 Java 类库的类增加薪的功能, 且不必对这些类重新进行设计. 比如, 只需在 Collection 接口中增加 default Stream<E> stream(), 相应的 Set 和 List 接口以及它们的子类都包含此方法, 不必为每个子类都重新copy这个方法.

这是一个折中的设计, 带来的问题就是为 Java 引入了多继承的问题. 接口可以继承接口, 类可以继承类和实现接口, 一旦继承的类和实现的接口中有相同签名的方法, 会出现什么样的状况? 本文将探讨各种情况的多继承, 以便能清楚的理解 Java 多继承的规则.

接口继承多个父接口

假定有三个接口 Interface A, Interface B, Interface C 继承关系:

interface C extends A, B
复制代码

A, B 拥有相同方法的名, 返回值, 参数的默认方法 default String(String name), 如果接口 C 没有 override 这个方法, 则编译出错.

interface A {
  default String say(String name){
    return "A: "+ name
  }
}

interface B {
  default String say(String name){
    return "B: "+ name
  }
}

interface C extends A, B {

}
复制代码

错误信息:

error: interface C inherits unrelated defaults for say(String) from types A and B
        static interface C extends A,B{
               ^
1 error
复制代码

我们可以在子接口 C 中覆盖( override )这个方法, 这样编译就不会出错了:

interface C extends A, B{
  default String say(String name){
    return "C: " + name
  }
}
复制代码
方法名相同; 返回值, 方法参数不同

注意在方法名相同, 但返回值, 方法参数不同, 情况有些区别:

public interface A {
    default String say(String name) {
        System.out.println("A: " + name);
        return "A: " + name;
    }
}

public interface B {
    default String say(int name) {
        System.out.println("B: " + name);
        return "B: " + name;
    }
}

public interface C extends A, B {
    @Override
    default String say(String name) {//参数为 String name
        System.out.println("C: " + name);
        return "C: " + name;
    }
}

public class D implements C {
}

public static void main(String[] args) {
  D d = new D();
  d.say(1);//System.out.println输出 B: 1
  d.say("1");//System.out.println输出 C: 1
}

//C还有另一种情况
public interface C extends A, B {
    @Override
    default String say(int name) {//参数为 int name
        System.out.println("C: " + name);
        return "C: " + name;
    }
}

public static void main(String[] args) {
  D d = new D();
  d.say(1);//System.out.println输出 B: 1
  d.say("1");//System.out.println输出 A: 1
}
复制代码

class D 有两种 say 方法:

方法名, 参数相同; 返回值不同

但在方法名, 参数相同, 返回值不同时 C 无法同时继承 A, B:

public interface A {
    default String say(String name) {
        System.out.println("A: " + name);
        return "A: " + name;
    }
}

public interface B {
    default int say(String name) {
        System.out.println("B: " + name);
        return 1;
    }
}
复制代码

方法名相同, 参数不相同, 返回值不同

方法名相同, 参数不相同, 返回值不同, C可以同时 Override A, B的方法

public interface A {
    default String say(String name) {
        System.out.println("A: " + name);
        return "A: " + name;
    }
}

public interface B {
    default int say(int name) {
        System.out.println("B: " + name);
        return name;
    }
}

public interface C extends A, B {
    @Override
    default String say(String name) {
        System.out.println("C: " + name);
        return name;
    }

    @Override
    default int say(int name) {
        System.out.println("C: " + name);
        return name;
    }
}

public class D implements C {
}

public static void main(String[] args) {
  D d = new D();
  d.say(1);//System.out.println输出 C: 1
  d.say("1");//System.out.println输出 C: 1
}
复制代码

接口的多层继承

假定有三个接口 Interface A, Interface B, Interface C 继承关系:

public interface A {
    default void say(String name) {
        System.out.println("A: " + name);
    }
}

public interface B extends A{
    @Override
    default void say(String name) {
        System.out.println("B: "+name);
    }
}

public interface C extends B{
    @Override
    default void say(String name) {
        System.out.println("C: " + name);
    }
}

public class D implements C {
}

public static void main(String[] args) {
  D d = new D();
  d.say("1");//System.out.println(); 输出 C: 1
}
复制代码

这个没有意外

类继承

public interface A {
    default void say(String name) {
        name = "A: " + name;
        System.out.println( name);
    }
}

public class B{
  public void say(String name) {
       name = "B: "+name;
       System.out.println(name);
   }
}

public class C extends B implements A {

}

public static void main(String[] args) {
  C c = new C();
  c.say("1");//System.out.println(); 输出 B: 1
}
复制代码

可以看出, 子类优先继承父类的方法

结论

其他更复杂的继承关系可以简化为以上的继承关系.

根据以上的例子, 可以得出以下的结论:

  • 类优先于接口. 如果一个子类继承的父类和实现的接口有相同的方法(方法名, 参数, 返回值都相同), 那么子类继承父类的方法

转载于:https://juejin.im/post/5a31d4b3f265da430c11d312

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值