• 【剧前爆米花--爪哇岛寻宝】面向对象的三大特性——封装、继承以及多态的详细剖析(上——继承)。


    作者:困了电视剧

    专栏:《JavaSE语法与底层详解》

    文章分布:这是一篇关于Java面向对象三大特性——继承的文章,在本篇文章中我会分享继承的一些基础语法以及类在继承时代码的底层逻辑和执行顺序。

    目录

    继承

    需求

    继承的定义及语法格式

    子类中访问父类成员

    super关键字

    super和this的区别

    子类构造方法

    子类的初始化过程——代码的执行顺序

    类从加载到实例化的大体流程

    举个栗子
    ​​​​​​​


    继承

    Java使用类这一引用类型对现实中的实体进行描述,通过类中的成员描述一个事物的属性,通过实例化对象来进行相关的操作,但是世界上的事物错综复杂,事物与事物之间很大可能会存在关联,在程序设计的时候就需要考虑。

    需求

    我们举一个日常生活中的栗子:人们在日常生活中可能会养一只猫或者是养一条狗,现在我需要用Java对他们进行描述:

    1. class Cat{
    2. String name;
    3. String sex;
    4. int age;
    5. public void sleep(){}
    6. public void eat(){
    7. System.out.println("吃猫粮");
    8. }
    9. public void speak(){
    10. System.out.println("喵");
    11. }
    12. }
    13. class Dog{
    14. String name;
    15. String sex;
    16. int age;
    17. public void sleep(){}
    18. public void eat(){
    19. System.out.println("吃狗粮");
    20. }
    21. public void speak(){
    22. System.out.println("汪");
    23. }
    24. }

    对于上述代码,我们可以清晰地看到,他们虽然是两个不同的事物,但是他们的属性相同,行为也有相同的地方,那我们定义两个类是不是就造成了代码的冗长,增加了出错的可能?

    为了解决这一问题Java 的研发团队引入了继承这一概念,专门用来进行共性抽取,实现代码复用。 

    继承的定义及语法格式

    继承(inheritance)机制:是面向对象程序设计使代码可以复用的最重要的手段,它允许程序员在保持原有类特 性 的基础上进行扩展,增加新功能,这样产生新的类,称派生类。继承呈现了面向对象程序设计的层次结构, 体现了 由简单到复杂的认知过程。继承主要解决的问题是:共性的抽取,实现代码复用。

    为了表示继承关系,Java中用extends关键字来表示类与类之间的继承。比如在上述栗子中,由于猫和狗都是动物,所以我可以写个动物类 ,让猫和狗都对其进行继承:

    1. class Animal{
    2. String name;
    3. String sex;
    4. int age;
    5. public void sleep(){}
    6. }
    7. class Cat extends Animal{
    8. public void eat(){
    9. System.out.println("吃猫粮");
    10. }
    11. public void speak(){
    12. System.out.println("喵");
    13. }
    14. }
    15. class Dog extends Animal{
    16. public void eat(){
    17. System.out.println("吃狗粮");
    18. }
    19. public void speak(){
    20. System.out.println("汪");
    21. }
    22. }

    对程序进行debug一下: 

    发现实例化出来的猫和狗对象中仍有相关的属性。

    其中:Animal类称为父类/基类或超类,Dog和Cat可以称为Animal的 子类/派生类,继承之后,子类可以复用父类中成员,子类在实现时只需关心自己新增加的成员即可

    从继承概念中可以看出继承最大的作用就是:实现代码复用,还有就是来实现多态

    子类中访问父类成员

    对于子类从父类中继承而来的变量有三种情况。

    1.子类中有的变量

    2.子类中没有,父类中有的变量

    3.子类中的与父类中的同名的变量

    这三种情况下子类中的变量应该如何进行访问,我们用一段代码来进行解释:

    1. public class Javabit_Code{
    2. public static void main(String[] args) {
    3. Derived der=new Derived();
    4. System.out.println(der.a);
    5. System.out.println(der.b);
    6. System.out.println(der.c);
    7. System.out.println(der.d);
    8. System.out.println("============");
    9. }
    10. }
    11. class Base{
    12. int a=10;
    13. int b=10;
    14. int d=10;
    15. }
    16. class Derived extends Base{
    17. int a=20;
    18. int c=20;
    19. double d=20;
    20. }

    运行结果如图所示:

     通过a和c输出的值可以看出不管父类中有没有,只要子类中有那就优先访问子类中的变量。

    由b可知,当子类中没有时会访问父类中的变量。

    由d可知,如果父类和子类中有同名的变量,当类型不同时也是优先访问子类的变量。

    super关键字

    我们一定会有一个需求,那就是虽然子类中有这个变量名的变量,但我仍需要父类中的这个变量的数据,但由于我的子类中有这个变量了,我无法对父类中的那个变量进行直接访问,为了解决这一问题,super关键字诞生了。

    1. class Base{
    2. int a=10;
    3. int b=10;
    4. int d=10;
    5. }
    6. class Derived extends Base{
    7. int a=super.a;
    8. int c=20;
    9. double d=20;
    10. }

     输出结果:

    注意:super关键字只是在代码层次上提高了代码的阅读性。在子类中代表继承过来的父类型的特征,不包含在子类对象中。通过super.data就能知道访问的是父类的data。 

    解释一下:子类在实例化的时候并不会实例化父类,这里的父类型的特征只是因为子类先执行了父类的构造方法(以自身意愿进行的初始化形式)进行了初始化,然后调用子类的构造方法在进行一次初始化。

    super和this的区别

    super不是引用类型,super中存储的不是内存地址,super指向的不是父类对象,super代表的是当前子类对象中的父类型特征。
    this是引用型变量,保存对象的内存地址(和super区别),super和this都需要依托对象的存在,才可以使用(所以static修饰的成员变量和成员方法都无法使用this和super)

    子类构造方法

    在子类构造方法中,并没有写任何关于基类构造的代码,但是在构造子类对象时,先执行基类的构造方法,然后执 行子类的构造方法,因为:子类对象中成员是有两部分组成的,基类继承下来的以及子类新增加的部分 。父子父子 肯定是先有父再有子,所以在构造子类对象时候 ,先要调用基类的构造方法,将从基类继承下来的成员构造完整 ,然后再调用子类自己的构造方法,将子类自己新增加的成员初始化完整

    1. class Base{
    2. int a=10;
    3. int b=10;
    4. int d=10;
    5. Base(){}
    6. Base(int a,int b){
    7. this.a=a;
    8. this.b=b;
    9. }
    10. }
    11. class Derived extends Base{
    12. int a=20;
    13. int b=20;
    14. int c=20;
    15. double d=20;
    16. Derived(){}//没有写super是因为父类中有无参的构造方法,实际上有super()只是省略了
    17. Derived(int a,int b,int c){
    18. super(a,b);
    19. this.c=c;
    20. }
    21. }

    子类的初始化过程——代码的执行顺序

    现在我们已知的类在加载和实例化对象的过程中,有静态代码块,实例代码块,构造方法和成员变量等,那么他们在这一过程中的执行顺序是什么?或者说一个类从被加载到实例化一个对象他都进行了那些步骤?

    1. public class Javabit_Code{
    2. public static void main(String[] args) {
    3. Derived der1=new Derived();
    4. System.out.println("==================");
    5. Derived der2=new Derived();
    6. }
    7. }
    8. class Number{
    9. public Number(){
    10. System.out.println("Number的构造方法");
    11. }
    12. }
    13. class Base{
    14. Number num=new Number();
    15. public Base(){
    16. System.out.println("父类的构造方法");
    17. }
    18. //实例代码块
    19. {
    20. System.out.println("父类的实例代码块");
    21. }
    22. //静态代码块
    23. static {
    24. System.out.println("父类的静态代码块");
    25. }
    26. }
    27. class Derived extends Base{
    28. public Derived(){
    29. System.out.println("子类的构造方法");
    30. }
    31. //实例代码块
    32. {
    33. System.out.println("子类的实例代码块");
    34. }
    35. //静态代码块
    36. static {
    37. System.out.println("子类的静态代码块");
    38. }
    39. }

     运行结果如图所示:

     由此我们可以得出结论:

    1、父类静态代码块优先于子类静态代码块执行,且是最早执行。

    2、父类实例代码块和父类构造方法紧接着执行。

    3、子类的实例代码块和子类构造方法紧接着再执行。

    4、第二次实例化子类对象时,父类和子类的静态代码块都将不会再执行。

    用父类的构造方法(super)初始化父类的成员变量,然后子类在 初始化自己独有的成员变量。

    类从加载到实例化的大体流程

    一个java文件被javac编译成.class字节码文件——》需要new一个对象的时候先检测类是否已经加载,没有加载则加载——》类加载并被放入jvm的同时会初始化静态变量,执行静态代码块——》为实例化的对象分配内存空间——》处理并发安全问题——》初始化所分配的空间即为成员变量默认初始化——》设置对象头信息——》先调用实例代码块在调用构造方法给对象中的各个成员赋值。

    举个栗子

    1. class X{
    2. Y y=new Y();
    3. public X(){
    4. System.out.print("X");
    5. }
    6. }
    7. class Y{
    8. public Y(){
    9. System.out.print("Y");
    10. }
    11. }
    12. public class Z extends X{
    13. Y y=new Y();
    14. public Z(){
    15. System.out.print("Z");
    16. }
    17. public static void main(String[] args) {
    18. new Z();
    19. }
    20. }

    这段代码的运行结果是YXYZ。

    执行顺序:new一个Z对象——》Z继承X——》先将X的成员构造完整——》调用了Y的构造方法输出了一个Y——》X的构造方法中也输出了一个X——》回到Z,将Z中的成员构造完整——》调用Y在输出一个Y(此时的Y对象为子类的Y对象,与从父类继承的Y对象同名但不相同)——》Z中的构造方法输出一个Z。

    以上就是我关于继承的讲解,如果对你有帮助还请三连,如果疏漏,欢迎大佬指正!

  • 相关阅读:
    TFT-eSPI 库在 ESP32 上的配置和使用(ESP32 for Arduino)
    快速查看Oracle数据库告警日志的存储位置
    【JAVASE】程序异常处理
    【代码】Android|判断asserts下的文件存在与否,以及普通文件存在与否
    15.变量的存储类别
    ASP.NET Core 6框架揭秘实例演示[30]:利用路由开发REST API
    bvh文件,人体骨骼重定向
    IDEA中pom文件出现灰色且有删除的线解决方案
    【SpringCloud】微服务技术栈入门2 - Nacos框架与Feign
    Ubuntu本地安装MySQL8.0以及常见问题设置方法
  • 原文地址:https://blog.csdn.net/m0_62815572/article/details/128057567