【Java基础】 为什么Java不支持多继承

简介: C++的多继承和Java的多实现

多继承是为了保证子类能够复用不同父类的方法,使用多继承会产生存在菱形继承的问题。C++使用虚继承的方式解决菱形继承问题。在现实生活中,我们真正想要使用多继承的情况并不多。因此在Java中并不允许多继承,但是Java可以通过以多接口的方式实现多继承的功能,即一个子类复用多个父类的方法。当接口中有同名方法时,子类必须重写同名方法。

此外,如果一个类继承了多个父类,那么势必会继承大量的属性和方法,这样会导致类的接口变得十分庞大,难以理解和维护。当尝试去修改父类时,会影响到多个子类,增加了代码的耦合度。

Java 8以前,接口中是不能有方法的实现的。所以一个类同时实现多个接口的话,也不会出现C++中的歧义问题。因为所有方法都没有方法体,真正的实现还是在子类中的。但是,Java 8中支持了默认函数(default method ),即接口中可以定义一个有方法体的方法了。

而又因为Java支持同时实现多个接口,这就相当于通过implements就可以从多个接口中继承到多个方法了,但是,Java8中为了避免菱形继承的问题,在实现的多个接口中如果有相同方法,就会要求该类必须重写这个方法。

扩展知识

菱形继承问题

假设我们有类B和类C,它们都继承了相同的类A。另外我们还有类D,类D通过多重继承机制继承了类B和类C

    类A
   /    \
  /      \
类B      类C
  \      /
   \    /
    类D

在上面这个结构中,类A是基类,类B和类C是派生类,而类D从类B和类C继承。如果类B和类C修改了来自A的某个属性或方法,类D在调用该属性或方法时,编译器或运行时环境就不清楚应该使用B的版本还是C的版本,形成了歧义。

C++中的菱形问题

下面是一个C++中的菱形问题例子:

#include <iostream>

class A {
   
public:
    virtual void doSomething() {
   
        std::cout << "Doing something in A\n";
    }
};

class B : public A {
   
public:
    void doSomething() override {
   
        std::cout << "Doing something in B\n";
    }
};

class C : public A {
   
public:
    void doSomething() override {
   
        std::cout << "Doing something in C\n";
    }
};

class D : public B, public C {
   
    // D现在从两个父类B和C继承了doSomething()方法
};

int main() {
   
    D d;
    // d.doSomething(); // 这里会引发编译错误,因为编译器不知道应该调用B的doSomething还是C的doSomething
    d.B::doSomething(); // 明确调用B中的doSomething
    d.C::doSomething(); // 明确调用C中的doSomething
    return 0;
}

在上面的代码中,类D继承自类B和类C,而这两个类都覆盖了来自类AdoSomething()方法。在类D的实例d中调用doSomething()方法时,编译器无法决定应该调用B的实现还是C的实现,因为存在二义性。为了解决这个问题,必须明确指出希望调用哪个父类的方法,如d.B::doSomething()d.C::doSomething()

Java中,这个问题通过不允许类多重继承来避免,但可以通过接口实现类似多重继承的效果。当然,如果接口中有相同的默认方法,也需要在实现类中明确指出使用哪个接口中的实现。

C++为了解决菱形继承问题,又引入了虚继承

C++中,虚继承是解决菱形问题(或钻石继承问题)的机制。通过虚继承,可以确保被多个类继承的基类只有一个共享的实例。

当两个类(如BC)从同一个基类(如A)虚继承时,无论这个基类被继承多少次,最终派生类(如D)中只包含一个基类A的实例。下面的C++代码示例展示了虚继承的使用:

#include <iostream>

class A {
   
public:
    int value;
    A() : value(1) {
   }
};

class B : virtual public A {
   
    // 使用virtual关键字进行虚继承
};

class C : virtual public A {
   
    // 使用virtual关键字进行虚继承
};

class D : public B, public C {
   
    // D从B和C继承,B和C都是从A虚继承而来
};

int main() {
   
    D d;
    d.value = 2; // 正确,无歧义,因为只有一个A的实例
    std::cout << d.value << std::endl; // 输出2

    B b;
    b.value = 3; // 正确,无歧义
    std::cout << b.value << std::endl; // 输出3

    C c;
    c.value = 4; // 正确,无歧义
    std::cout << c.value << std::endl; // 输出4

    return 0;
}

在这个例子中,class Bclass C都是通过关键字virtualclass A那里继承而来的。这意味着在class D中,不管通过B还是C的路径,A只有一个实例,从而解决了因多个实例导致的歧义问题。

虚继承通常涉及到一个额外的开销,因为编译器需要维护虚基类的信息,以确保在运行时可以正确地构造和定位虚基类的实例。因此,只有在需要解决菱形问题时才应该使用虚继承。

因为支持多继承,引入了菱形继承问题,又因为要解决菱形继承问题,引入了虚继承。而经过分析,人们发现我们其实真正想要使用多继承的情况并不多。

所以,在 Java 中,不允许“声明多继承”,即一个类不允许继承多个父类。但是 Java 允许“实现多继承”,即一个类可以实现多个接口,一个接口也可以继承多个父接口。由于接口只允许有方法声明而不允许有方法实现(Java 8之前),这就避免了 C++ 中多继承的歧义问题。

Java 8中的多继承

Java不支持多继承,但是是支持多实现的,也就是说,同一个类可以同时实现多个接口。

我们知道,在Java 8以前,接口中是不能有方法的实现的。所以一个类同时实现多个接口的话,也不会出现C++中的歧义问题。因为所有方法都没有方法体,真正的实现还是在子类中的。

那么问题来了。

Java 8中支持了默认函数(default method ),即接口中可以定义一个有方法体的方法了。

public interface Pet {
   

    public default void eat(){
   
        System.out.println("Pet Is Eating");
    }
}

而又因为Java支持同时实现多个接口,这就相当于通过implements就可以从多个接口中继承到多个方法了,这不就是变相支持了多继承么。

那么,Java是怎么解决菱形继承问题的呢?我们再定义一个哺乳动物接口,也定义一个eat方法。

public interface Mammal {
   

    public default void eat(){
   
        System.out.println("Mammal Is Eating");
    }
}

然后定义一个Cat,让他分别实现两个接口:

public class Cat implements Pet,Mammal {
   

}

这时候,编译期会报错:

error: class Cat inherits unrelated defaults for eat() from types Mammal and Pet

这时候,就要求Cat类中,必须重写eat()方法。

public class Cat implements Pet,Mammal {
   
    @Override
    public void eat() {
   
        System.out.println("Cat Is Eating");
    }
}

所以可以看到,Java并没有帮我们解决多继承的歧义问题,而是把这个问题留给开发人员,通过重写方法的方式自己解决。

耦合度增加

由于Java不允许多重继承,在这里使用一个假设性的代码示例来解释如果Java允许多重继承,会发生什么情况。

假设我们有两个父类ClassAClassB,它们都有大量的方法和属性:

class ClassA {
   
    public void methodA1() {
    /* ... */ }
    public void methodA2() {
    /* ... */ }
    // ... 更多方法

    public int propertyA1;
    public int propertyA2;
    // ... 更多属性
}

class ClassB {
   
    public void methodB1() {
    /* ... */ }
    public void methodB2() {
    /* ... */ }
    // ... 更多方法

    public int propertyB1;
    public int propertyB2;
    // ... 更多属性
}

现在,我们创建一个类ClassC,它假设性地从ClassAClassB中继承:

// 假设的多重继承,在Java中实际上是不允许的
class ClassC extends ClassA, ClassB {
   
    public void methodC() {
   
        // ClassC 的特定方法
    }
}

在这个假设的多重继承场景中,ClassC会继承来自ClassAClassB的所有方法和属性。这导致了几个问题:

  1. 接口庞大
    ClassC的接口变得非常庞大,它包含了ClassAClassB所有的方法和属性。这使得ClassC非常复杂,难以理解和使用。

  2. 维护困难
    由于ClassC依赖于两个父类,任何对ClassAClassB的修改都可能影响到ClassC。如果父类中的方法签名发生了变化,或者某些属性被重命名或删除,ClassC都需要做出相应的更新。

  3. 冲突解决
    如果ClassAClassB中有同名的方法或属性,ClassC需要有一种机制来解决这些命名冲突。在C++中,这可以通过指定父类的作用域来解决,但Java避免这种问题的方式是根本不允许多重继承。

class ClassC extends ClassA, ClassB {
   
    public void methodA1() {
   
        // 需要解决方法冲突,决定使用 ClassA 的 methodA1
        super(ClassA).methodA1();
    }
    // 假设这样的语法存在,在Java中实际上并不支持
}

这种情况下的代码耦合度非常高,因为ClassC对两个父类都有依赖,修改任何一个父类都可能需要对ClassC进行修改。这样的设计使得系统的可维护性降低,同时也降低了代码的稳定性。

在真实的Java编程中,我们通常使用接口来实现类似多重继承的效果,并通过设计模式如组合(Composition)和接口分离(Interface Segregation)来降低类的复杂性和耦合度。

相关文章
|
2天前
|
Java
Java 面向对象编程的三大法宝:封装、继承与多态
本文介绍了Java面向对象编程中的三大核心概念:封装、继承和多态。
40 15
|
2天前
|
存储 移动开发 算法
【潜意识Java】Java基础教程:从零开始的学习之旅
本文介绍了 Java 编程语言的基础知识,涵盖从简介、程序结构到面向对象编程的核心概念。首先,Java 是一种高级、跨平台的面向对象语言,支持“一次编写,到处运行”。接着,文章详细讲解了 Java 程序的基本结构,包括包声明、导入语句、类声明和 main 方法。随后,深入探讨了基础语法,如数据类型、变量、控制结构、方法和数组。此外,还介绍了面向对象编程的关键概念,例如类与对象、继承和多态。最后,针对常见的编程错误提供了调试技巧,并总结了学习 Java 的重要性和方法。适合初学者逐步掌握 Java 编程。
12 1
|
2月前
|
Java
在Java中,接口之间可以继承吗?
接口继承是一种重要的机制,它允许一个接口从另一个或多个接口继承方法和常量。
158 1
|
2月前
|
Java 大数据 API
14天Java基础学习——第1天:Java入门和环境搭建
本文介绍了Java的基础知识,包括Java的简介、历史和应用领域。详细讲解了如何安装JDK并配置环境变量,以及如何使用IntelliJ IDEA创建和运行Java项目。通过示例代码“HelloWorld.java”,展示了从编写到运行的全过程。适合初学者快速入门Java编程。
|
3月前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
52 3
|
3月前
|
Java
在Java多线程编程中,实现Runnable接口通常优于继承Thread类
【10月更文挑战第20天】在Java多线程编程中,实现Runnable接口通常优于继承Thread类。原因包括:1) Java只支持单继承,实现接口不受此限制;2) Runnable接口便于代码复用和线程池管理;3) 分离任务与线程,提高灵活性。因此,实现Runnable接口是更佳选择。
75 2
|
3月前
|
Java
Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口
【10月更文挑战第20天】《JAVA多线程深度解析:线程的创建之路》介绍了Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口。文章详细讲解了每种方式的实现方法、优缺点及适用场景,帮助读者更好地理解和掌握多线程编程技术,为复杂任务的高效处理奠定基础。
55 2
|
3月前
|
Java 开发者
Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点
【10月更文挑战第20天】Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点,重点解析为何实现Runnable接口更具灵活性、资源共享及易于管理的优势。
58 1
|
3月前
|
Java 测试技术 编译器
Java零基础-继承详解!
【10月更文挑战第4天】Java零基础教学篇,手把手实践教学!
54 2
|
3月前
|
Java 编译器
在Java中,关于final、static关键字与方法的重写和继承【易错点】
在Java中,关于final、static关键字与方法的重写和继承【易错点】
42 5