• JavaSE - 继承


    目录

    1. 什么是继承

    2. 父类成员的访问

    2.1 子类中访问父类的成员变量

    2.2 子类中访问父类的成员方法

    2.3 重载和重写的区别 

    3. super关键字

    3.1 super的用法

    3.2 super和this的区别

    4. 子类构造方法:给子类的成员变量赋值

    4.1 父类中是显式定义无参或者默认的无参构造方法

    4.1.1 子类中也是显式定义无参或者默认的无参构造方法

    ​4.1.2 子类中是有参构造方法

    4.2 父类中是有参构造方法

    4.2.1 子类中是显式定义无参构造方法

    4.2.2 子类中是有参构造方法

    5.如何给子类的成员变量赋值

    5.1 子类和父类不存在同名成员变量

    5.2 子类和父类存在同名成员变量

    6. 代码块

    7. protected 关键字

    8. final 关键字

    9. 继承与组合


    1. 什么是继承

    子类:又叫派生类

    父类:又叫基类/超类

    继承:就是对共性的抽取,从而达到对代码的复用【重复使用】

    在Java中,只能继承一个类,不能多继承

    当子类继承父类之后,就会把父类中的成员变量和成员方法全部继承到子类中

    私有的成员也可以被继承,但是不能被访问。要想被访问父类的成员中要提供公开的方法

    子类继承父类之后,必须要新添加自己特有的成员,体现出与基类的不同,否则就没有必要继承了

    2. 父类成员的访问

    先去子类中找,找到则访问,找不到再去父类中找,找到则访问,否则编译报错。

    子类父类都有此成员,就近原则优先访问子类

    2.1 子类中访问父类的成员变量

    子类继承父类中的所有成员变量和成员方法,父类中的puiblic修饰的成员子类都能直接访问

    当父类和子类中拥有同名的成员变量时,遵循就近原则,会优先访问子类自己的

    非要访问父类的,前面要加个【super点号】

     

    2.2 子类中访问父类的成员方法

    和访问父类中的成员变量一样

    当父类和子类中拥有同名的成员方法时,遵循就近原则,会优先访问子类自己的

    非要访问父类的,前面要加个【super点号】

    2.3 重载和重写的区别 

    重载:可以发生在一个类里的多个方法之间,也可以发生在继承当中

    • 1. 方法名称必须一样
    • 2. 方法的参数必须不一样(个数,数据类型,不同数据类型的顺序  这三个中,满足一个不同就算参数不一样)
    • 3. 返回值不做要求,可以相同也可以不同

    重写:在子类中把父类本身有的方法重新写一遍

    • 1. 方法名称必须一样
    • 2. 方法的参数必须一样(个数,数据类型,不同数据类型的顺序  都要一样)
    • 3. 返回值也必须一样
    • 4. 修饰符可以不一样,但是有要求:子类的访问权限大于等于父类的访问权限且修饰符不可以是private,private修饰的方法是不可以重写的。

    访问权限由大到小:【public > protected > default > private 

    具体见多态。

    3. super关键字

    由于设计不好,或者因场景需要,子类和父类中可能会存在相同名称的成员,如果想要在子类方法中访问父类同名成员,需要用到super关键字。

    3.1 super的用法

    1. super.data;【访问父类的普通(非静态)成员变量】

    2. super.func();【调用父类的普通(非静态)成员方法】

    3. super();【调用父类的构造方法

    不用super去调用静态的成员变量或方法,静态的是直接通过类名去调用的。

    super只能在非静态方法中使用静态方法中不能有super,不能有this静态方法中只能是静态的。

    这样就会有一个面试问题:super和this的区别是什么?

    3.2 super和this的区别

     

    同:

    • 1. 都是Java中的关键字
    • 2. 只能在类的非静态方法中使用,用来访问非静态成员方法和字段
    • 3. 在构造方法中调用时,必须是构造方法中的第一条语句,并且不能同时存在

    异:

    • 1. this是当前对象的引用,当前对象即调用实例方法的对象,super相当于是子类对象中从父类继承下来部分成员的引用【说super是父类的引用是完全错误的,父类根本就没new对象,super只是子类对象中从父类继承下来部分成员的引用
    • 2. 在非静态成员方法中,this用来访问本类的方法和属性,super用来访问父类继承下来的方法和属性
    • 3. 在构造方法中:this(...)用于调用本类构造方法,super(...)用于调用父类构造方法,两种调用不能同时在构造方法中出现
    • 4. 构造方法中一定会存在super(...)的调用,用户没有写编译器也会增加,但是this(...)用户不写则没有

    4. 子类构造方法:给子类的成员变量赋值

    先给从父类中继承的成员变量赋值再给子类自己新增的成员变量赋值

    实例化子类对象时,JVM会将对象中的成员初始化成初始值,接着会调用子类构造方法。super();的作用是:调用父类的构造方法。所以会先给从父类中继承的成员变量赋值,再给子类自己新增的成员变量赋值。

    注意:

    1. 虽然执行了父类的构造方法,但是并没有生成父类的对象,这里只是帮助子类给从父类继承过来的属性赋值。

    2. 在子类构造方法中,super(...)调用父类构造时,必须是子类构造方法中第一条语句

    3. super(...)不能this同时出现在一个构造方法中,因为它们都要放在构造方法的第一条语句,会形成冲突,不知道执行哪条语句。

    4.1 父类中是显式定义无参或者默认的无参构造方法

    那么在子类构造方法第一行默认会有隐含的super()调用【调用父类构造方法

    4.1.1 子类中也是显式定义无参或者默认的无参构造方法

    只要是无参构造方法,无论子类和父类有没有提供,或是一方提供一方没提供,子类构造方法第一行有没有super();

    只要没有,编译器都会帮我们生成一份默认的无参构造方法,没有的编译器都会帮我们补全

    最终会补全成下面这样:【子类父类都有无参构造方法,子类的无参构造方法第一行有super();】

    4.1.2 子类中是有参构造方法

    子类中提供有参构造方法父类中是显式定义无参或者默认的无参构造方法,

    那么编译器将不会帮我们在子类中生成无参构造方法,若父类中不是显式定义无参,编译器会帮我们在父类中生成一份默认的无参构造方法,子类有参构造方法第一行默认有隐含的super();

    最终会补全成下面这样:【子类有参父类无参,子类的有参构造方法第一行有super();】

    4.2 父类中是有参构造方法

    那么子类构造方法第一行不会有隐含的super()调用【调用父类构造方法需要自己写

    无论子类中的构造方法有参还是无参也都得自己写,都得是显性的。

    4.2.1 子类中是显式定义无参构造方法

     ​​​​​​

    4.2.2 子类中是有参构造方法

    5.如何给子类的成员变量赋值

    成员变量包括:从父类继承的和子类自己新增的

    5.1 子类和父类不存在同名成员变量

    1. 定义时就直接赋值

    2. 在main函数中通过【引用.成员变量】进行赋值

    3. 在子类中使用成员方法

    4. 在子类中使用构造方法

    有参数的构造方法、构造方法中调用本类当中的其他构造方法)

    【使用有参数的构造方法进行赋值最好用】

    1. class Animals{
    2. public String name;
    3. public int age;
    4. public String sex = "男";// 1. 定义时就直接赋值
    5. }
    6. public class Dog extends Animals {
    7. public boolean silly;
    8. /**
    9. * 3. 使用成员方法
    10. */
    11. public void setDog(int age,boolean silly){
    12. this.age = age;
    13. this.silly = silly;
    14. }
    15. @Override
    16. public String toString() {
    17. return name+" "+age+" "+sex+" "+silly;
    18. }
    19. public static void main(String[] args) {
    20. Dog dog = new Dog();
    21. /**
    22. * 2. 在main函数中通过【引用.成员变量】进行赋值
    23. */
    24. dog.name = "大黄";
    25. dog.setDog(10,true);
    26. System.out.println(dog);
    27. }
    28. }

    4.1 使用有参数的构造方法

    4.2 使用无参数的构造方法,并调用本类当中的其他构造方法

    4.3 使用有参数的构造方法,并调用本类当中的其他构造方法

    5.2 子类和父类存在同名成员变量

    相同成员变量中如何分别继承的父类成员变量子类自己的成员变量赋值?

    使用构造方法进行赋值时,父类是有参构造方法参数是同名的成员变量时,是给继承的父类成员变量赋值;父类是默认的无参构造方法时,说明根本没有访问到父类的成员变量,所以是给子类自己的成员变量赋值。

    (!!!)就近原则,this在父类的构造方法中,就是给父类的成员变量赋值;this在子类的构造方法中,就是给子类的成员变量赋值。

    使用构造方法给继承的父类成员变量子类自己的成员变量赋值对比

    (1)有参数的构造方法:

    (2) 构造方法中调用本类当中的其他构造方法:

     

    给子类自己的成员变量赋值:

    1. 定义时就直接赋值

    2. 在子类中使用成员方法(因为就近原则,方法中的成员变量都是子类自己的)

    3. 使用构造方法

    有参数的构造方法、构造方法中调用本类当中的其他构造方法)

    给继承的父类成员变量赋值:

    1. 定义时就直接赋值

    2. 在子类使用成员方法【成员变量前要加super点号】

    3. 在父类使用成员方法(因为就近原则,方法中的成员变量都是父类自己的)

    4. 使用构造方法

    有参数的构造方法、构造方法中调用本类当中的其他构造方法)

    1. class Animals{
    2. public String name = "小黑";// 1. 定义时就直接赋值
    3. public int age;
    4. public String sex;
    5. }
    6. public class Dog extends Animals {
    7. public String name;
    8. public int age;
    9. /**
    10. * 2. 在子类中使用成员方法【成员变量前面要加super点号】
    11. */
    12. public void setDog(String name,int age,String sex) {
    13. super.name = name;
    14. super.age = age;
    15. super.sex = sex;
    16. }
    17. @Override
    18. public String toString() {
    19. return sex+" "+name+" "+age+" "+" "+super.name+" "+super.age;
    20. }
    21. public static void main(String[] args) {
    22. Dog dog = new Dog();
    23. dog.setDog("hello",18,"男");
    24. System.out.println(dog);
    25. }
    26. }
    1. class Animals{
    2. public String name;
    3. public int age;
    4. public String sex;
    5. /**
    6. * 3. 在父类中使用成员方法
    7. */
    8. public void setDog(String name,int age){
    9. this.name = name;
    10. this.age = age;
    11. }
    12. }
    13. public class Dog extends Animals {
    14. public String name;
    15. public int age;
    16. @Override
    17. public String toString() {
    18. return sex+" 子类中的:"+name+" "+age+" "+"父类中的:"+super.name+" "+super.age;
    19. }
    20. public static void main(String[] args) {
    21. Dog dog = new Dog();
    22. dog.setDog("小灰",2);
    23. System.out.println(dog);
    24. }
    25. }

    6. 代码块

    静态代码块类加载的时候被执行,它不依赖于对象,且只会执行一次

    实例代码块无参数的构造方法只有在创建对象时才会执行,且创建几个对象执行几次

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

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

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

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

    顺序:父类中的静态代码块,子类中的静态代码块,父类中的实例代码块,父类中的构造代码块,子类中的实例代码块,子类中的构造代码块

    1. class Animals{
    2. public String name;
    3. public int age;
    4. public String sex;
    5. static{
    6. System.out.println("Animals::static{}");
    7. }
    8. {
    9. System.out.println("Animals::{}");
    10. }
    11. public Animals(){
    12. System.out.println("Animals()");
    13. }
    14. }
    15. class Cat extends Animals{
    16. static{
    17. System.out.println("Cat::static{}");
    18. }
    19. public Cat(){
    20. System.out.println("Cat()");
    21. }
    22. {
    23. System.out.println("Cat::{}");
    24. }
    25. }
    26. public class Dog extends Animals {
    27. public String name;
    28. public int age;
    29. static{
    30. System.out.println("Dog::static{}");
    31. }
    32. public Dog(){
    33. System.out.println("Dog()");
    34. }
    35. {
    36. System.out.println("Dog::{}");
    37. }
    38. public static void main(String[] args) {
    39. Dog dog = new Dog();
    40. Cat cat = new Cat();
    41. }
    42. }

    7. protected 关键字

    可访问范围:同一个包中,不同包中的它的子类

    8. final 关键字

    final关键可以用来修饰变量、成员方法以及类

    1. 修饰变量或字段,表示常量(即不能修改)【public final int a = 10;】

    2. 修饰方法:就变成了密封方法,表示该方法不能被重写

    3. 修饰类:就变成了密封类,表示此类不能被继承【public final class A{}】

    9. 继承与组合

    组合也能达到代码复用的效果。组合并没有涉及到特殊的语法, 仅仅是将一个类的实例作为另外一个类的字段。

    继承表示对象之间是is-a的关系

    class Dog extends Animal  -》  Dog is a Animal

    组合表示对象之间是a part of/has-a的关系:

    学生和老师是学校的一部分

    学校有学生和老师

    1. class Teacher{
    2. }
    3. class Student{
    4. }
    5. class School{
    6. //一个学校有很多老师和很多学生组成
    7. public Teacher[] teacher;
    8. public Student[] student;
    9. }

  • 相关阅读:
    CC57 链表内指定区间反转
    运维管理系统,人性化操作体验
    有什么可以代替Calendar的吗?
    Java高级特性-泛型类
    系统数据数据和信息
    AM@导数求导法则
    数据结构与算法(二):数组与链表
    使用Python 3脚本自动化Harbor镜像复制
    val的准确率高于train是过拟合吗
    JDBC执行Oracle的Sql脚本注意细节
  • 原文地址:https://blog.csdn.net/m0_61731585/article/details/126223256