extends继承与final关键字

目录

extends继承

什么时候用继承?

继承的格式?

继承的特点

子类可以继承父类的哪些呢?

是否可以继承父类的构造方法呢?

是否可以继承成员变量?

是否可以继承成员方法?

继承中成员变量的访问特点

继承中成员方法的访问特点

方法的重写

方法重写注意事项和要求

继承中构造方法的访问特点

final关键字


extends继承

在Java中,extends关键字用于实现继承关系。通过使用extends关键字,一个类可以继承另一个类的属性和方法。

继承允许我们通过创建更具体的子类来扩展和改进现有的类。子类可以访问父类中的公共方法和属性,并可以添加自己的方法和属性。通过继承,我们能够实现代码的重用性,并可以更好地组织和管理程序的结构。

什么时候用继承?

 当类与类之间,存在相同(共性)的内容,并满足子类是父类中的一种,就可以考虑使用继承,来优化代码

继承的格式?

public class 子类extends 父类{}

继承的特点

 Java只支持单继承,不支持多继承,但支持多层继承。
多层继承:子类A继承父类B,父类B可以继承父类
每一个类都直接或者间接的继承于0bject 类

子类可以继承父类的哪些呢?

是否可以继承父类的构造方法呢

下面我们来一探究竟

先来个父类

class Fu{
    String name;
    int age;
    public Fu(){

    }
    public  Fu(String name,int age){
        this.name=name;
        this.age=age;

    }
}

再来个子类

class Zi extends Fu{

}

下面我们就来测试一下

可以看到带参构造创建的子类对象报错了

idea建议我们创建一个构造方法

那上面的空参构造为什么没有报错呢?

一个类中如果没有构造方法,虚拟机会自动的给我们添加一个默认的空参构造

是否可以继承成员变量?

成员变量可以被继承。在面向对象编程中,子类继承父类时可以继承父类的成员变量。当子类继承父类时,子类将会拥有父类的所有成员变量和成员方法,包括私有的成员变量。子类对象可以直接访问继承下来的成员变量,或者通过父类的公用方法来间接访问。需要注意的是,如果子类需要对继承的成员变量进行修改,可以使用继承下来的成员变量进行赋值,或者通过继承过来的成员方法来操作。

是否可以继承成员方法?

虚方法表中的可以,否则不行

虚方法表(Virtual Method Table,VMT)是一种用于实现面向对象编程语言中的多态性的机制。在这些编程语言中,对象的方法可以被子类重写以实现不同的行为。虚方法表通过一个表格来记录对象的方法,这个表格中的每一项指向实际执行的方法。当使用对象的方法时,编译器会根据对象的类型在虚方法表中查找相应的方法,并调用它。

虚方法表是面向对象编程语言中的一个重要概念,它使得多态性得以实现。通过将方法的调用与方法的实际实现解耦,虚方法表允许程序在运行时动态地确定要调用的方法。

在很多面向对象编程语言中,每个对象都有一个指向其所属类的虚方法表的指针。这个指针指向一个特定类的虚方法表,在表中,每个方法都被分配了一个唯一的索引。当对象的方法被调用时,程序会根据对象的类型查找相应索引对应的方法,并调用它。

虚方法表的使用方便了对象的方法的重写和动态绑定,使得多态性得以实现。它是实现面向对象编程语言中的多态性的一种常用机制。

继承中成员变量的访问特点

就近原则:谁离我近,我就用谁

先在局部位置找,本类成员位置找,父类成员位置找,逐级往上。

我们在代码中看看

先写个Fu类

public class Fu {
    String name = "Fu";
}

再写个Zi类

public class Zi extends Fu {
    String name = "Zi";

    public void ziShow() {
        String name="ziShow";

        System.out.println(name);
    }
}

这个输出的结果会是什么呢 ?

下面我们进行测试

public class Test4 {
    public static void main(String[] args) {
        Zi zi = new Zi();
        zi.ziShow();

    }
}

可以看到,输出结果为ziShow

离它最近的是ziShow

如果我将Zi类中的String name="ziShow";删掉

这样答案又会是什么呢?

根据就近原则,他就会打印Zi

如果我将Zi类中的String name="Zi"也删掉

答案又会是什么?

它就会去Fu类中去寻找最近的

如果Fu类中也没有的话它就会报错

那我如果想将这三个name都输出出来该怎么办呢?

我们只需要这样输出就好了

System.out.println(name);//从局部位置开始往上找
System.out.println(this.name);//从本类成员位置开始往上找
System.out.println(super.name);//从父类成员位置开始往上找

继承中成员方法的访问特点

直接调用满足就近原则:谁离我近,我就用谁
super调用,直接访问父类

我们来写个例子看看

先写个Person类

public class Person {
    public void eat(){
        System.out.println("吃米饭,吃菜");
    }
    public void drink(){
        System.out.println("喝水");
    }
}

再写个Student类

public class Student extends Person{
    public void lunch(){
        eat();
        drink();
    }
}

测试一下

public class Test5 {
    public static void main(String[] args) {
        Student student = new Student();
        student.lunch();
    }
}

结果为

先在本类中査看eat和drink方法,就会调用子类的,如果没有,就会调用从父类中继承下来的eat和drink方法

我们再Student类中加上super

super是直接去父类中寻找

如果现在我将Student类中的代码改为

public class Student extends Person{
    public void lunch(){
        this.eat();
        this.drink();
        super.eat();
        super.drink();
    }
    public void eat(){
        System.out.println("吃牛排");
    }
    public void drink(){
        System.out.println("喝可乐");
    }
}

person类中还是

public class Person {
    public void eat(){
        System.out.println("吃米饭,吃菜");
    }
    public void drink(){
        System.out.println("喝水");
    }
}

 又会输出什么呢?

this先在本类中査看eat和drink方法,一看有就直接调用子类的方法了

super直接从父类中找eat和drink方法


到这里就要介绍方法的重写了

方法的重写

 当父类的方法不能满足子类现在的需求时,需要进行方法重写
书写格式
在继承体系中,子类出现了和父类中一模一样的方法声明,我们就称子类这个方法是重写的方法

@Override重写注解
1.@Override是放在重写后的方法上,校验子类重写时语法是否正确

2.加上注解后如果有红色波浪线,表示语法错误。

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

public class Person {
    public void eat(){
        System.out.println("吃米饭,吃菜");
    }
    public void drink(){
        System.out.println("喝水");
    }
}
public class Student extends Person{
    public void lunch(){
        this.eat();
        this.drink();
        super.eat();
        super.drink();
    }
    @Override
    public void eat(){
        System.out.println("吃牛排");
    }
    @Override
    public void drink(){
        System.out.println("喝可乐");
    }
}

方法重写注意事项和要求

  1. 重写方法的名称、形参列表必须与父类中的一致。
  2. 子类重写父类方法时,访问权限子类必须大于等于父类(空着不写<protected<public)
  3. 子类重写父类方法时,返回值类型子类必须小于等于父类
  4. 建议:重写的方法尽量和父类保持一致。
  5. 私有方法不能被重写。
  6. 子类不能重写父类的静态方法,如果重写会报错的。

5、与6、可以理解为只有被添加到虚方法表中的方法才能被重写 

继承中构造方法的访问特点

  • 父类中的构造方法不会被子类继承。
  • 子类中所有的构造方法默认先访问父类中的无参构造,再执行自己

为什么呢?

  • 子类在初始化的时候,有可能会使用到父类中的数据,如果父类没有完成初始化,子类将无法使用父类的数据。
  • 子类初始化之前,一定要调用父类构造方法先完成父类数据空间的初始化。

那是如何调用父类构造方法的?

  • 子类构造方法的第一行语句默认都是:super(),不写也存在,且必须在第一行。
  • 如果想调用父类有参构造,必须手动写super进行调用

 

 这个Test运行结果是什么呢?

子类中所有的构造方法默认先访问父类中的无参构造,再执行自己。子类构造方法的第一行语句默认都是:super(),不写也存在,且必须在第一行。


如果想调用父类有参构造,必须手动写super进行调用

final关键字

final关键字是用来表示一个变量、方法或类不可被修改的。


final修饰方法:表明该方法是最终方法,不能被重写

final修饰类:表明该类是最终类,不能被继承

final修饰变量:叫做常量,只能被赋值一次


final 修改基本数据类型:记录的值不能发生改变


final 修饰引用数据类型:记录的地址值不能发生改变,内部的属性值还是可以改变的

报错了,地址值不能被修改,我们来看看内部的属性值呢

可以看到引用数据类型的数据被修改了


这期就到这里啦,感谢阅读,继续努力呀!!!

  • 30
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

努力敲代码的小火龙

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值