2024-03-13  阅读(2)
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。 本文链接:https://www.skjava.com/mianshi/baodian/detail/2808832612

回答

Java 不支持多重继承的主要原因是为了避免复杂性和简化设计,特别是为了避免“菱形问题”。在多重继承的情况下,如果一个类继承自两个具有相同方法签名的父类,子类将不清楚应该继承哪个父类的方法,这会导致歧义和设计上的复杂性。

虽然 java 不支持多重继承,但是我们可以通过使用接口或者接口的默认方法来间接实现类似“多重继承”的功能。

扩展

菱形问题

菱形问题是多重继承可能引发的一个典型问题。假如有类 BC ,他们继承 A,然后又有一个类 D 通过多重继承的方式继承 BC,那么他们之间的继承关系就形成了一个菱形:

假如 A 定义了一个 doSomething()BC 两个子类都重写了 doSomething(),如果这个时候 D 调用 doSomething(),那么编译器或在运行时就会面临一个问题:应该调用 BdoSomething() ,还是 CdoSomething() ?如果只有 B 重写了 doSomething() 方法,那么它是否应该直接调用 A 的原始版本,还是 C 的呢?

所以这会产生一个歧义的问题。出了歧义问题,还会面临维护问题,如果父类中的方法在未来发生变化,可能会在不经意间破坏子类的行为,使得维护和理解代码变得更加困难。

为了解决这个菱形问题,不同的编程语言采取不同的策略:

  • C++:通过虚继承(Virtual Inheritance)允许多重继承,但要求开发者显式解决潜在的冲突。
  • Java:不允许使用多重继承。

在 Java 中如何实现“多重继承”

虽然 Java 不支持多重继承,但是我们可以通过接口和接口的默认方法来间接实现类似“多重继承”的功能。

使用接口实现多重继承

Java 允许一个类实现多个接口,同时提供这些接口中所有方法的实现。这种允许我们定义可以由单个类实现的多个方法签名,从而在一定程度上模拟多重继承。

interface InterfaceA {
    void methodA();
}

interface InterfaceB {
    void methodB();
}

class MyClass implements InterfaceA, InterfaceB {
    public void methodA() {
        // 实现methodA
    }

    public void methodB() {
        // 实现methodB
    }
}

使用默认方法实现接口的“多重继承”

Java 引入默认方法,允许接口中有方法的具体实现。这就意味着一个接口可以提供某个方法的实现,而实现该接口的类则可以直接使用这个方法,或者根据需要重写它。这样,接口不仅可以定义方法的签名,还可以提供实现,使得通过接口模拟多重继承变得更加强大。

interface InterfaceA {
    void methodA();
    
    default void defaultMethodA() {
        // ...
    }
}

interface InterfaceB {
   void methodB();

    default void defaultMethodB() {
        // ...
    }
}

class MyClass implements InterfaceA, InterfaceB {
    public void methodA() {
      // ...
    }
    
    public void methodB() {
      // ...
    }

    // 重写 defaultMethodB
    public void defaultMethodB() {
        // ...
    }
}

但是使用默认方法会有一个问题。假如 InterfaceA 和 InterfaceB 都提供默认方法 defaultMethod()

public interface InterfaceA {
    default void defaultMethod(){
        System.out.println("InterfaceA - defaultMethod");
    }
}

public interface InterfaceB {
    default void defaultMethod(){
        System.out.println("InterfaceB - defaultMethod");
    }
}

public class MyClass implements InterfaceA,InterfaceB{
}

结果如下:

这个提示告诉我们:当一个类实现的多个接口中存在签名相同的默认方法时,Java编译器将无法决定应该继承哪个接口中的方法实现,我们需要重写这个默认方法来消除这个歧义:

public class MyClass implements InterfaceA,InterfaceB{
    @Override
    public void defaultMethod() {
        InterfaceA.super.defaultMethod();
    }
}

Java 面试宝典是大明哥全力打造的 Java 精品面试题,它是一份靠谱、强大、详细、经典的 Java 后端面试宝典。它不仅仅只是一道道面试题,而是一套完整的 Java 知识体系,一套你 Java 知识点的扫盲贴。

它的内容包括:

  • 大厂真题:Java 面试宝典里面的题目都是最近几年的高频的大厂面试真题。
  • 原创内容:Java 面试宝典内容全部都是大明哥原创,内容全面且通俗易懂,回答部分可以直接作为面试回答内容。
  • 持续更新:一次购买,永久有效。大明哥会持续更新 3+ 年,累计更新 1000+,宝典会不断迭代更新,保证最新、最全面。
  • 覆盖全面:本宝典累计更新 1000+,从 Java 入门到 Java 架构的高频面试题,实现 360° 全覆盖。
  • 不止面试:内容包含面试题解析、内容详解、知识扩展,它不仅仅只是一份面试题,更是一套完整的 Java 知识体系。
  • 宝典详情:https://www.yuque.com/chenssy/sike-java/xvlo920axlp7sf4k
  • 宝典总览:https://www.yuque.com/chenssy/sike-java/yogsehzntzgp4ly1
  • 宝典进展:https://www.yuque.com/chenssy/sike-java/en9ned7loo47z5aw

目前 Java 面试宝典累计更新 400+ 道,总字数 42w+。大明哥还在持续更新中,下图是大明哥在 2024-12 月份的更新情况:

想了解详情的小伙伴,扫描下面二维码加大明哥微信【daming091】咨询

同时,大明哥也整理一套目前市面最常见的热点面试题。微信搜[大明哥聊 Java]或扫描下方二维码关注大明哥的原创公众号[大明哥聊 Java] ,回复【面试题】 即可免费领取。

阅读全文