Java 继承详解

在这里插入图片描述

个人主页:熬夜磕代码丶
作品专栏: java se
我变秃了,也变强了
给大家介绍一款程序员必备刷题平台——牛客网
点击注册一起刷题收获大厂offer吧

一、什么是继承?

多个类中存在相同属性和行为时,将这些内容抽取到单独一个类中,那么多个类无需再定义这些属性和行为,只要继承那个类即可。
多个类可以称为子类,单独这个类称为父类、超类或者基类。子类可以直接访问父类中的非私有的属性和行为。通过 extends 关键字让类与类之间产生继承关系。

class Dog extends Animal
    //Dog 是子类  //Animal是父类

在这里插入图片描述

二、为什么要继承?

2.1 继承的方式

当多个类存在相同的属性和方法时,在每次描述时都要写一遍,这样代码的冗余率很高,但是如果有继承机制,我们可以把这些相同的属性和方法抽取出来,书写一个父类,当子类需要的时候,直接继承就不用对这部分进行书写,只需要进行补充。

代码如下(示例):

class Animal {
    
    
    public String name;
    public int age;
    public void eat() {
    
    
        System.out.println(name+"正在吃饭! ");
    }
}
class Dog extends Animal {
    
    
    public void bark() {
    
    
        System.out.println(name+"汪汪叫! ");
    }
}
class Cat extends Animal {
    
    
    public void mew() {
    
    
        System.out.println(name+"喵喵叫! ");
    }
}

在这里猫和狗都有姓名,年龄,吃东西的特性,我们可以将这些共性,写成一个Animal类,直接继承然后写入特有的bark和mew方法即可。

继承的优点:
1.提高代码的复用性。
2.让类与类之间产生了关系,是多态的前提。

2.2 继承的特点

在这里插入图片描述
1.Java只支持单继承,不支持多继承。
2.Java支持多层(重)继承(继承体系)。
3.继承关系一般不希望超过三层,如果超过那么就考虑重构了.

三、继承后如何访问?

3.1访问父类成员变量

代码如下(示例):

//不存在同名的情况
class Father {
    
    
    int a;
    int b;
}
class Son extends Father {
    
    
    int c;
    public void func() {
    
    
        a = 10;
        b = 20;
        c = 30;
    }
}

在这里a,b直接访问的时父类继承下来的a,b,可以进行赋值

//父类和子类成员变量同名
class Father {
    
    
    int a;
    int b;
}
class Son extends Father {
    
    
    int a;
    int b;
    int c;
    public void func() {
    
    
        a = 10;
        b = 20;
        c = 30;
    }
}

在这里a,b访问的是子类中的a,b

总结:
1.如果访问的成员变量如果子类有,优先访问子类的。
2.如果子类中无,则访问父类中的,如果父类也没有,则编译失败.
3.如果访问子类父类中都有,则优先访问子类的,就近原则.

3.2访问父类成员方法

//父类子类成员方法名不同
class Father {
    
    
    public void func1() {
    
    
        System.out.println("func1");
    }
}    
class Son extends Father {
    
    
    public void func2() {
    
    
        System.out.println("func2");
    }
    public void func3() {
    
    
     func1();
     func2();
    }
}

在func3()中访问的func1()是父类的func1()

class Father {
    
    
    public void func1() {
    
    
        System.out.println("Father: func1");
    }
}
class Son extends Father {
    
    
    public void func1() {
    
    
        System.out.println("Son: func1");
    }
    public void func2() {
    
    
        System.out.println("Son: func2");
    }
    public void func3() {
    
    
        func1();
        func2();
    }
}

在func3()中访问的func1()是子类的func1()

总结:
1.子类与父类方法名不同时,优先在子类找,如果子类找不到,则去父类找,如果父类找不到,则编译错误。
2.子类与父类同名方法时,如果父类和子类同名方法的参数不同,则根据调用选择合适的参数进行访问,如果没有合适的则报错误.

四、super的使用

如果子类中存在与父类中相同的成员时,那如何在子类中访问父类相同名称的成员呢?
可以使用super关键字进行访问.

4.1 访问父类的成员

class Father {
    
    
    int a;
    int b;
}
class Son extends Father {
    
    
    int a;
    int b;
    int c;
    public void func() {
    
    
        super.a = 10;
        super.b = 20;
        c = 30;
    }
}

4.2 访问父类的方法

class Father {
    
    
    public void func1() {
    
    
        System.out.println("Father: func1");
    }
}
class Son extends Father {
    
    
    public void func1() {
    
    
        System.out.println("Son: func1");
    }
    public void func2() {
    
    
        System.out.println("Son: func2");
    }
    public void func3() {
    
    
        super.func1();
        func2();
    }
}

注意:
1.super只能在非静态方法中使用
2.在子类方法中,访问父类的成员变量和方法

4.3 子类构造方法

在生成子类对象时,会先调用父类的构造方法,在去执行子类的构造方法.一般没有写出来,系统会默认写入.

class Father {
    
    
    public Father() {
    
    
        System.out.println("Father!");
    }
}
class Son extends Father {
    
    
    public Son() {
    
    
        //super();
        System.out.println("Son!");
    }

    public static void main(String[] args) {
    
    
        Son son = new Son();
    }
}

在这里插入图片描述
这里验证了我们所说的,在实例子类对象时,会先调用父类的实例方法,然后在调用我们子类的实例方法,在子类的构造方法默认有一个super()方法去调用父类的构造方法,如果我们自己写的父类的构造方法是带参数的,我们但自己在子类构造方法第一行写super(参数)不然会编译报错.

4.4 super和this

1.super是一个关键字,代表父类的存储空间标识。(可以理解为父亲的引用)
2.super和this的用法相似。
3.this代表对象的引用(谁调用就代表谁);
4.super代表当前子类对父类的引用。
5.super();和this();都是在构造函数的第一行,不能同时出现。
不同点
1.成员变量

this.变量        本类的
super.变量       父类的

2.构造方法

this(...)        本类的
super(...)       父类的

3.成员方法

this.方法名()        本类的    
super.方法名()       父类的

4.5 继承代码块执行顺序

1.父类静态代码块优先执行,然后子类静态代码块执行,静态代码块只执行一次
2.在实例子类对象时,父类实例代码块和父类构造方法紧接着执行.
3.子类的实例代码块和子类构造方法再执行

五、组合

5.1 final关键字

1.final修饰变量表示常量,不能再修改(常量书写默认为单词的大写)

final int COUNT = 0;
COUNT = 10;//编译错误

2.final修饰类表示该类不能在被继承

final class Father {
    
    

}
class Son extends Father {
    
    

}

在这里插入图片描述
3.final修饰的方法不能够重写

5.2 组合

组合表示的是:对象之间存在has的关系,其中一个类是其他几个类的组合.
在这里插入图片描述

class Tire {
    
    

}
class Engine {
    
    

}
class VehicleSystem {
    
    

}
class Car {
    
    
    private Tire tire;
    private Engine engine;
    private VehicleSystem vehicleSystem;
}
class Ferrari extends Car {
    
    
    
}

猜你喜欢

转载自blog.csdn.net/buhuisuanfa/article/details/126250119
今日推荐