继承详细说明

概述

Java中提供一个关键字extends,用这个关键字,我们可以让一个类和另一个类建立起父子关系。

例如:public class Student extends People {}

Student称为子类(派生类),People称为父类(基类 或超类)。

使用继承的好处

当子类继承父类后,就可以直接使用父类公共的属性和方法了。

提高代码复用性,减少代码冗余,增强类的功能扩展性。

案例练习:

请阅读以下代码存在的问题,并使用继承这个技术进行优化。

 解决方案:把相同的属性和行为抽离出来,构成父类,子类继承即可

继承的设计规范

子类们相同特征(共性属性,共性方法)放在父类中定义,子类独有的的属性和行为应该定义在子类自己里面。

为什么? 如果子类的独有属性、行为定义在父类中,会导致其它子类也会得到这些属性和行为,这不符合面向对象逻辑。

内存运行原理?

下面的描述与上面有冲突。

在Java中,对象的内存布局并不直接包含方法引用。方法引用实际上是与类的元数据信息相关联的,而不是与单个对象实例相关联的。当我们创建一个对象时,该对象包含了对其类(包括父类和子类)的引用,而不是直接包含方法引用。

当我们讨论方法的调用时,实际上是在讨论如何通过对象的类引用找到正确的方法并执行它。这个过程涉及到Java虚拟机(JVM)的方法分派机制,它根据方法的签名和访问控制来确定应该调用哪个方法。

父类方法引用和子类方法引用

在Java中,子类继承了父类的方法。这意味着子类对象可以调用在父类中定义的方法。当子类需要覆盖(override)父类的方法时,子类会提供一个新的方法实现,具有与父类方法相同的签名(方法名和参数列表)。

当一个子类对象调用一个方法时,JVM会首先检查这个方法是否在子类中定义。如果子类定义了这个方法(即覆盖了父类的方法),则调用子类中的实现。如果子类没有覆盖该方法,则JVM会继续在父类中查找该方法,并调用父类中的实现。

方法分派

JVM使用两种主要的方法分派机制来确定应该调用哪个方法:静态分派和动态分派。

  • 静态分派:也称为编译时多态,发生在编译时期。它根据方法签名(方法名和参数类型)来确定应该调用哪个方法。重载(overloading)就是静态分派的一个例子。

  • 动态分派:也称为运行时多态,发生在运行时期。它基于对象的实际类型来确定应该调用哪个方法。重写(overriding)就是动态分派的一个例子。当一个子类对象调用一个被重写的方法时,JVM会根据对象的实际类型来调用相应的方法实现。

父类方法和子类方法的区分

在运行时,JVM通过对象的类引用和方法分派机制来区分应该调用父类方法还是子类方法。如果子类覆盖了父类的方法,那么子类对象调用该方法时将执行子类中的实现。如果子类没有覆盖该方法,或者调用的是父类特有的方法,那么将执行父类中的实现。

总结来说,虽然子类对象包含了对其父类的引用,但方法引用不是直接存储在对象内存中的。方法调用的过程涉及到JVM的方法分派机制,它根据对象的实际类型和方法的覆盖情况来确定应该调用哪个方法实现。

子类对象在内存中的布局

  • 子类成员变量:这些是子类自己定义的成员变量,它们存储在子类对象空间的开始部分。
  • 父类成员变量:这些是父类定义的成员变量,包括从父类的父类继承下来的成员变量和父类自身定义的成员变量。它们紧随子类成员变量之后存储。
  • 对象头:对象头包含了对象的元数据信息,如类元数据指针、锁信息、哈希码等。这部分信息对于JVM的垃圾收集、同步等机制非常重要。

继承的特点

子类可以继承父类的属性和行为,但是子类不能继承父类的构造器。

Java是单继承模式:一个类只能继承一个直接父类。

Java不支持多继承、但是支持多层继承。

Java中所有的类都是Object类的子类。

1、子类是否可以继承父类的构造器?

不可以的,子类有自己的构造器,父类构造器用于初始化父类对象。

2、子类是否可以继承父类的私有成员? 可以的,只是不能直接访问。

3、子类是否可以继承父类的静态成员?

有争议的知识点。

子类可以直接使用父类的静态成员(共享)

但个人认为:子类不能继承父类的静态成员。(共享并非继承) 

4、Java为何不支持多继承,请看如下反证法

public class 父类A {     public void method(){         System.out.println("复习数学");     } }

public class 父类B {     public void method(){         System.out.println("复习语文");     } }

public class 子类C extends 父类A , 父类B{    

        public static void main(String[] args) {              

                子类 z = new 子类();        

                z.method();   // 复习啥?出现二义性,听哪个爸爸的呢?java懵了!因此不支持多继承    

} }

继承后:成员变量、成员方法的访问特点

在子类方法中访问成员(成员变量、成员方法)满足:就近原则

        先子类局部范围找

        然后子类成员范围找

        然后父类成员范围找,如果父类范围还没有找到则报错。

如果子父类中,出现了重名的成员,会优先使用子类的,此时如果一定要在子类中使用父类的怎么办?

        可以通过super关键字,指定访问父类的成员。格式:super.父类成员变量/父类成员方法

继承后:方法重写

在继承体系中,子类出现了和父类中一模一样的方法声明,我们就称子类这个方法是重写的方法。

方法重写的应用场景

当子类需要父类的功能,但父类的该功能不完全满足自己的需求时。

子类可以重写父类中的方法。

案例演示:

旧手机的功能只能是基本的打电话,发信息

新手机的功能需要能够:基本的打电话下支持视频通话。基本的发信息下支持发送语音和图片。

public class Phone {
    public void call(){
        System.out.println("打电话开始~~~");
    }

    public void sendMessage(){
        System.out.println("发送短信开始~~~");
    }
}
public class NewPhone extends Phone{
    /**
      方法重写了
     */
    @Override
    public void call() {
        super.call();
        System.out.println("支持视频通话~~~");
    }

    /**
     方法重写了
     */
    @Override
    public void sendMessage() {
        super.sendMessage();
        System.out.println("支持发送图片和视频~~~");
    }
}

 @Override重写注解

@Override是放在重写后的方法上,作为重写是否正确的校验注解。

加上该注解后如果重写错误,编译阶段会出现错误提示。

建议重写方法都加@Override注解,代码安全,优雅!

方法重写注意事项和要求

重写方法的名称、形参列表必须与被重写方法的名称和参数列表一致。

私有方法不能被重写。

子类重写父类方法时,访问权限必须大于或者等于父类 (暂时了解 :缺省 < protected < public) 。

子类不能重写父类的静态方法,如果重写会报错的。

继承后:子类构造器的特点

子类中所有的构造器默认都会先访问父类中无参的构造器,再执行自己。

为什么?

子类在初始化的时候,有可能会使用到父类中的数据,如果父类没有完成初始化,子类将无法使用父类的数据。

子类初始化之前,一定要调用父类构造器先完成父类数据空间的初始化。

怎么调用父类构造器的?

子类构造器的第一行语句默认都是:super(),不写也存在。

继承后:子类构造器访问父类有参构造器

super调用父类有参数构造器的作用:

初始化继承自父类的数据。

如果父类中没有无参数构造器,只有有参构造器,会出现什么现象呢?

会报错,因为子类默认是调用父类无参构造器的。

如何解决?

子类构造器中可以通过书写 super(…),手动调用父类的有参数构造器

this、super使用总结

this:代表本类对象的引用;super:代表父类存储空间的标识。

案例

需求:

在学员信息登记系统中,后台创建对象封装数据的时候如果用户没有输入学校,则默认使用“黑马培训中心”。

如果用户输入了学校则使用用户输入的学校信息。 

public class Student {
    private String name;
    private String schoolName;

    public Student() {
    }

    public Student(String name) {
        // 借用兄弟构造器!
        this(name, "黑马培训中心");
    }


    public Student(String name, String schoolName) {
        this.name = name;
        this.schoolName = schoolName;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getSchoolName() {
        return schoolName;
    }

    public void setSchoolName(String schoolName) {
        this.schoolName = schoolName;
    }
}

this(...)和super(…)使用注意点:

子类通过 this (...)去调用本类的其他构造器,本类其他构造器会通过 super 去调用父类的构造器,最终还是会调用父类构造器的。

注意:this(…) super(…) 都只能放在构造器的第一行,所以二者不能共存在同一个构造器中。

  • 22
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C++中的棱形继承(diamond inheritance)是一种多重继承的形式,其中一个派生类同时继承自两个或更多个基类,而这些基类又共同继承自一个公共基类。这种继承关系形成了一个类似棱形的结构,因此得名为棱形继承。 下面我将详细说明棱形继承的特点和使用方法。 首先,让我们考虑下面这个示例: ```cpp class A { public: void funcA() { cout << "A::funcA()" << endl; } }; class B : public A { public: void funcB() { cout << "B::funcB()" << endl; } }; class C : public A { public: void funcC() { cout << "C::funcC()" << endl; } }; class D : public B, public C { public: void funcD() { cout << "D::funcD()" << endl; } }; ``` 在上面的代码中,我们定义了四个类:`A`、`B`、`C`和`D`。类`B`和类`C`都公有地继承自类`A`,而类`D`则同时继承自类`B`和类`C`。 这种继承关系形成了一个棱形结构,如下所示: ``` A / \ B C \ / D ``` 棱形继承的特点如下: 1. 类`D`继承自类`B`和类`C`,因此它同时拥有了类`B`和类`C`中的成员函数和成员变量。 2. 如果类`B`和类`C`中有相同的成员函数或成员变量,那么在类`D`中调用这些成员时,会产生二义性。编译器无法确定应该调用哪个基类中的成员。 3. 为了解决二义性问题,我们可以使用作用域解析运算符(`::`)来指定调用哪个基类中的成员。 下面是一个示例代码,演示了如何使用棱形继承: ```cpp int main() { D obj; obj.funcA(); // 调用 A::funcA() obj.funcB(); // 调用 B::funcB() obj.funcC(); // 调用 C::funcC() obj.B::funcA(); // 调用 B::funcA() obj.C::funcA(); // 调用 C::funcA() return 0; } ``` 在上面的代码中,我们创建了一个类`D`的对象`obj`,并调用了它继承自不同基类的成员函数。由于存在二义性问题,我们使用作用域解析运算符指定了调用哪个基类的成员函数。 希望这个例子能帮助你理解C++中的棱形继承。如果你还有其他问题,请随时提问!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值