美文网首页
Java基础系列15-面向对象之继承

Java基础系列15-面向对象之继承

作者: 只是甲 | 来源:发表于2021-08-17 10:15 被阅读0次

    一.继承概述

    继承的概述:

    1. 多个类中存在相同的属性和行为时,将这些内容抽取到单独一个类中,那么多个类无需在定义这些属性和行为,只要继承那个类(extends)即可.
    2. 单独的这个类称为父类,基类或者叫超类,多个类可以称为子类或者派生类.
    3. 有了继承以后,我们定义一个类的时候,可以在一个已经存在的类的基础上,还可以定义自己的新成员.

    实现继承的方式:

    1. 通过extends关键字可以实现类与类的继承
    2. 格式:public class 子类名 extends 父类名{}

    1.1 继承的好处和弊端

    继承的好处:

    1. 提高了代码的复用性,多个类相同的成员可以放到同一个类中;
    2. 提高了代码的维护性,如果功能的代码需要修改,修改一处即可;
    3. 让类与类之间产生了关系,是多态的前提,好处的第三点同时也是继承的弊端。

    继承的弊端:
    类与类之间产生了关系,让类的耦合性增强了

    1.2 Java中继承的特点

    1. Java中只支持单继承,不支持多继承
      1)一个类只能有一个父类,不可以有多个父类
      public class Son extends Father{} // ok
      public class Son extends Father,GrandFather // Error

    2. Java中类支持多层继承(继承体系)
      Public class GrandFather{}
      Public class Father extends GrandFather{}
      Public class Son extends Father{}

    1.3 Java继承中成员变量的特点

    A.局部变量
      a.定义位置:定义在局部范围中,如:函数内,语句内等;
      b.初始值:无,先定义,赋值后才能使用;
      c.调用方式:- - -
      d.作用域: 只在所属的区域有效;
      e.物理存储: 存储在 虚拟机栈(JVM Stack) 中(栈内存分为:虚拟机栈和本地方法栈);
    变量名 和 值 都在虚拟机栈(JVM Stack)的栈帧(Stack Frame) 的 局部变量表(Local Variable Table) 中;
      f.生命周期:与方法共存亡,随着方法的调用而存在,随着方法调用完毕自动回收释放;
    
    B.成员变量(别名:实例变量)
      a.定义位置:在类中,方法外;
      b.初始值:有默认初始化值;
      c.调用方式:对象调用;
      d.作用域:在整个类中都可以被访问;
      e.物理存储:
        对象的实例存储在 堆内存(Heap) 中。堆内存速度慢、成本低、空间较大。这个堆内存会有一个内存地址;
        对象的引用存储在 ==虚拟机栈(JVM Stack)中
      f.生命周期:与对象共存亡,随着对象的创建而存在,随着对象被回收而释放;
    
    C.静态变量(别名:类变量)
      a.定义位置:由staic修饰的,在类中,方法外;
      b.初始值:有默认初始化值;
      c.调用方式:对象调用,类名调用;
      d.作用域:全局中都能使用、被所有对象所共享;
      e.物理存储:存储在 方法区(Method Area) 的 静态域(Static Field) 中;
      f.生命周期:与类共存亡,随着类的加载而存在,随着类的消失而消失。
    

    成员变量名称不一样,使用的时候非常简单
    成员变量名称一样的情况:
     在子类中访问变量:(就近原则)
      在方法的局部范围找,如果有就使用
      在子类的成员范围找,如果有就使用
      在父类的成员范围找,如果有就使用
      如果还找不到 就报错

    image.png

    1.4 继承案例

    1.4.1 父子类案例

    接下来,我们创建一个父子类,父类是人类的类,子类是学生类,然后学生类继承人类 类。

    代码:
    jicheng1类

    package Java_study;
    /**
     * 
     * @author    只是甲
     * @date      2021-6-24
     * @remark    继承案例1, 人类 类
     *
     */
    public class jicheng1 {
        private String name;
        private int age;
        
        /**
         * @remark 无参构造
         */
        public jicheng1() {
            
        }
        
        /**
         * @remark  全参构造
         * @param   name
         * @param   age
         */
        public jicheng1(String name, int age) {
            this.name = name;
            this.age = age;
        }
        
        public String getName() {
            return name;
        }
        
        public void setName(String name) {
            this.name = name;
        }
        
        public int getAge() {
            return age;
        }
        
        public void setAge(int age) {
            this.age = age;
        }
    
    }
    

    student_new类

    package Java_study;
    
    /**
     * @author   只是甲
     * @date     2021-06-24
     * @remark   学生类,继承人类类
     */
    public class student_new extends jicheng1 {
        public void teach() {
            System.out.println("老师要好好讲课");
        }
    }
    

    jicheng_test

    package Java_study;
    
    /**
     * @author   只是甲
     * @date     2021-06-24
     * @remark   测试类
     */
    public class jicheng_test{
        public static void main(String[] args) {
            //创建对象
            student_new student1 = new student_new();
            
            //调用父类方法
            student1.setName("杜兰特");
            student1.setAge(32);
            System.out.println("学生姓名:" + student1.getName());
            System.out.println("学生年龄:" + student1.getAge());
            
            //调用子类方法
            student1.teach();
        }
    }
    

    测试记录:

    学生姓名:杜兰特
    学生年龄:32
    老师要好好讲课
    

    1.4.2 多重继承案例

    接下来是一个多重继承的,祖父、父亲、儿子三个类。

    代码:
    grandFather类:

    package Java_study;
    
    /**
     * 
     * @author  Administrator
     * @date    2021-06-24
     * @remark  多重继承祖父类
     */
    public class grandFather {
        public void grandFatherSay() {
            System.out.println("爷爷都是从孙子熬过来的");
        }
    }
    
    

    father类

    package Java_study;
    
    /**
     * 
     * @author  Administrator
     * @date    2021-06-24
     * @remark  多重继承父类
     */
    public class father extends grandFather {
        public void fatherSay() {
            System.out.println("爸爸都是从儿子走过来的");
        }
    
    }
    

    son类

    package Java_study;
    
    /**
     * 
     * @author  Administrator
     * @date    2021-06-24
     * @remark  多重继承子类
     */
    public class son extends father {
        public static void main(String[] args) {
            father son = new father();
            son.fatherSay();
            son.grandFatherSay();
        }
    
    }
    

    测试记录:

    爸爸都是从儿子走过来的
    爷爷都是从孙子熬过来的
    

    二. super关键字以及继承中的方法重写

    2.1 super关键字的概述和使用

    super的用法和this很像
      this代表本类对象的引用
      super代表父类存储空间的标识(可以理解为父类对象引用)
      super()且放在第一行 ; 目的是在初始化当前对象时,先保证了父类对象先初始化 。

    用法(this和super均可如下使用)
      访问成员变量
        this.成员变量
        super.成员变量
      访问构造方法
        this(…)
        super(…)
      访问成员方法
        this.成员方法()
        super.成员方法()

    构造方法中this()或者super()要放在第一行

    1. 在构造函数中,如果你不指定 构造器之间的调用关系 ,那么编译器会给你加上 super()且放在第一行 ; 目的是在初始化当前对象时,先保证了父类对象先初始化 。所以,你指定了构造函数间的调用,那么this()必须在第一行,以保证在执行任何动作前,对象已经完成了初始化。
    2. 构造函数只能被构造函数调用,因为对象只会初始化一次。
      this()和super()这样的方法被称为构造方法,顾名思义,他的作用就是在JVM堆中构建出一个指定类型的对象,如果你调用了两个这种形式的方法,岂不是代表着构建出了两个对象。
    3. 同理,为了避免构建出两个对象这种问题的出现,Java在编译时对这种情况做了强校验, 用户不能再同一个方法内调用多次this()或super(),同时为了避免对对象本身进行操作时,对象本身还未构建成功(也就找不到对应对象),所以对this()或super()的调用只能在构造方法中的第一行实现,防止异常。
    4. 在普通的成员方法中,如果调用super()或者this(),你是想要重新创建一个对象吗?抱歉Java为了保证自身对象的合理性,不允许你做这样的操作。

    代码:
    super1

    package Java_study;
    
    /**
     * 
     * @author  只是甲
     * @date    2021-06-25
     * @remark  人类父类   super关键字概述
     *
     */
    public class super1 {
        //为了演示案例的方便,这里我们使用public修饰了成员变量,实际开发中,修饰符使用private
        //年龄
        public int age = 45;//c:在父类的成员范围找,如果有就使用
    }
    

    super2

    package Java_study;
    
    /**
     * 
     * @author  只是甲
     * @date    2021-06-25
     * @remark  人类子类   super关键字概述
     *
     */
    public class super2 extends super1 {
        // 年龄
        public int age = 20;
        
        public void printAge() {
            int age = 10;
            //我要访问局部范围的age?//10
            System.out.println(age);
            //我要访问成员范围的age
            System.out.println(this.age);
            //我要访问父类成员范围的age? //45
            System.out.println(super.age);
        }
    }
    

    super3

    package Java_study;
    
    /**
     * 
     * @author  只是甲
     * @date    2021-06-25
     * @remark  人类测试类   super关键字概述
     *
     */
    public class super3 {
        public static void main(String[] args) {
            super2 s = new super2();
            //s.show();
            //10
            //20
            //45
            s.printAge();
        }
    
    }
    

    2.2 Java继承中构造方法的特点

    子类所有构造方法都默认访问父类的空参数的构造方法
    为什么呢?
      因为子类会继承父类中的数据,可能还会使用父类的数据,所以,子类初始化之前,一定要先完成父类数据的初始化
      每一个构造方法的第一条默认语句都是super

    如果父类中没有构造方法,该怎么办呢?
      在父类中加一个无参的构造方法
      通过使用super关键字去显示的调用父类的带参构造方法
      通过这里我们发现第一种解决方案最简单,所以,建议我们自定义类的时候永远自己给出无参构造方法

    代码:
    jicheng_gouzao1

    package Java_study;
    /**
     * 
     * @author   只是甲
     * @date     2021-06-25
     * @remark   人类父类   Java继承中构造方法的访问特点
     *
     */
    
    public class jicheng_gouzao1 {
        /**
         * @remark  Father无参构造方法
         */
        public jicheng_gouzao1() {
            System.out.println("Father无参构造方法");
        }
        
        /**
         * @remark  Father带参构造方法
         * @param   name  姓名
         */
        public jicheng_gouzao1(String name) {
            System.out.println("Father带参构造方法");
            System.out.println("father:" + name);
        }
    
    }
    

    jicheng_gouzao2

    package Java_study;
    
    /**
     * 
     * @author   只是甲
     * @date     2021-06-25
     * @remark   人类子类   Java继承中构造方法的访问特点
     *
     */
    public class jicheng_gouzao2 extends jicheng_gouzao1 {
        String name;
        public jicheng_gouzao2() {
            //super();
            System.out.println("Son无参构造方法");
        }
        
        public jicheng_gouzao2(String name) {
            super( name);
            //this();
            System.out.println("Sone带参构造方法");
            System.out.println("son:" + name);
        }
    
    }
    

    jicheng_gouzao3

    package Java_study;
    
    /**
     * 
     * @author   只是甲
     * @date     2021-06-25
     * @remark   人类测试类   Java继承中构造方法的访问特点
     *
     */
    public class jicheng_gouzao3 {
        public static void main(String[] args) {
            /*
            Father无参构造方法
            Son无参构造方法
            */
            jicheng_gouzao2 s1 = new jicheng_gouzao2();
            
            System.out.println("----------");
            
            /*
            Father带参构造方法
            father:杜兰特
            Son带参构造方法
            son:杜兰特
            */
            jicheng_gouzao2 s2 = new jicheng_gouzao2("杜兰特");
        }
    
    }
    

    测试记录:

    Father无参构造方法
    Son无参构造方法
    ----------
    Father带参构造方法
    father:杜兰特
    Sone带参构造方法
    son:杜兰特
    

    2.3 Java继承中成员方法的特点

    通过子类对象去访问一个方法

    1. 首先在子类中找
    2. 然后在父类中找
    3. 如果还是没有就会报错

    代码:
    jicheng_chengyuan1

    package Java_study;
    
    /**
     * 
     * @author  只是甲
     * @date    2021-06-25
     * @remark  人类父类  Java继承中成员方法的访问特点
     *
     */
    public class jicheng_chengyuan1 {
        public void method() {
            System.out.println("Father method");
        }
        
        public void show() {
            System.out.println("Father show");
        }
    }
    

    jicheng_chengyuan2

    package Java_study;
    
    /**
     * 
     * @author  只是甲
     * @date    2021-06-25
     * @remark  人类子类  Java继承中成员方法的访问特点
     *
     */
    public class jicheng_chengyuan2 extends jicheng_chengyuan1 {
        @Override
        public void method() {
            super.method();
        }
        
        /**
         * @remark  重写父类方法
         */
        @Override
        public void show() {
            System.out.println("Son show");
        }
    }
    

    jicheng_chengyuan3

    package Java_study;
    
    /**
     * 
     * @author  只是甲
     * @date    2021-06-25
     * @remark  人类测试类  Java继承中成员方法的访问特点
     *
     */
    public class jicheng_chengyuan3 {
        public static void main(String[] args) {
            jicheng_chengyuan2 s = new jicheng_chengyuan2();
            
            //直接调用父类的method方法:Father method
            s.method();
            
            //重写了父类的show方法:Son show
            s.show();
        }
    
    }
    

    测试记录:

    Father method
    Son show
    

    2.4 方法重写的概述和使用

    方法重写的概述:
      方法重写:子类中出现了和父类中一摸一样的方法声明

    方法重写的应用:
      当子类需要父类的功能,而功能主体子类有自己特有的内容时,可以重写中的方法,这样重写父类中的方法
      这样,即沿袭了父类的功能,又定义了子类特有的内容

    方法重写的注意事项:

    1. 注解
      @Override表明该方法的重写父类的方法
    2. 方法重写的注意事项
      1)父类中私有方法不能被重写
      2)子类重写父类方法时,访问权限不能更低
      3)子类重写父类方法时,建议访问权限一摸一样

    具体案例可以参考上一节2.3。

    参考:

    1. https://blog.csdn.net/qq_43529621/article/details/115209518

    相关文章

      网友评论

          本文标题:Java基础系列15-面向对象之继承

          本文链接:https://www.haomeiwen.com/subject/cbwbyltx.html