12 继承

简介: 12 继承

配套视频教程

本文B站配套视频教程

为什么使用继承


image.png


image.png


使用继承

编写父类

class Pet { 
        //公共的属性和方法
}

编写子类

class Dog extends Pet { 
        //子类特有的属性和方法
}

class Penguin extends Pet { 
}

子类访问父类成员super

使用super关键字,super代表父类对象

访问父类构造方法

super();    --super调用构造方法时,只能是第一句

super(name);

访问父类属性

super.name;

访问父类方法

super.print();

不能被子类继承的父类成员

  • private成员
  • 构造方法

protected

可以修饰属性和方法

本类、同包、子类可以访问

访问修饰符小结


image.png


多重继承关系的初始化顺序

1.父类属性

2.父类构造方法

3.子类属性

4.子类构造方法

class Person {
    String name="李光";// 姓名
    public Person() {
        // super();//写不写该语句,效果一样
        System.out.println("execute Person()");
    }
    public Person(String name) {
        this.name = name;
        System.out.println("execute Person(name)");
    }
}
class Student extends Person {
    String school="蓝翔";// 学校
    public Student() {
        // super();//写不写该语句,效果一样
        System.out.println("execute Student() ");
    }
    public Student(String name, String school) {
        super(name); // 显示调用了父类有参构造方法,将不执行无参构造方法
        this.school = school;
        System.out.println("execute Student(name,school)");
    }
}
class PostGraduate extends Student {
    String guide;// 导师
    public PostGraduate() {
        // super();//写不写该语句,效果一样
        System.out.println("execute PostGraduate()");
    }
    public PostGraduate(String name, String school, String guide) {
        super(name, school);
        this.guide = guide;
        System.out.println("execute PostGraduate(name, school, guide)");
    }
}
class TestInherit {
    public static void main(String[] args) {
        PostGraduate pgdt=null;
        pgdt = new PostGraduate();
        System.out.println();
        pgdt=new PostGraduate("刘小光","北京大学","王老师");
    }
}

继承条件下构造方法的调用规则如下:

  • 如果子类的构造方法中没有通过super显式调用父类的有参构造方法,也没有通过this显式调用自身的其他构造方法,则系统会默认先调用父类的无参构造方法。在这种情况下,写不写“super();”语句,效果是一样的。
  • 如果子类的构造方法中通过super显式调用父类的有参构造方法,那将执行父类相应构造方法,而不执行父类无参构造方法。
  • 如果子类的构造方法中通过this显式调用自身的其他构造方法,在相应构造方法中应用以上两条规则。
  • 特别注意的是,如果存在多级继承关系,在创建一个子类对象时,以上规则会多次向更高一级父类应用,一直到执行顶级父类Object类的无参构造方法为止。

阅读代码,思考运行结果

class Car { 
    private int site = 4;  //座位数
    Car(){
        System.out.println ("载客量是"+site+"人);
    }
    public void setSite(int site){
        this.site = site;
    }
    void print(){
        System.out.print("载客量是"+site+"人");
    }
}
class Bus extends Car { 
    Bus(int site){
        setSite(site);
    }
}
public static void main(String[] args) { 
    Bus bus = new Bus(20);
    bus.print();
}

方法重写

使用继承之后


image.png


调用父类的print()方法,不能显示Dog的strain信息和Peguin的sex信息


image.png


以上效果如何实现

**子类重写父类方法 ** @Override

方法重写需注意的问题


1.构造方法因为不能被继承,所以不能被重写;

方法重写的规则

  • 方法名相同
  • 参数列表相同
  • 返回值类型相同或者是其子类;
  • 访问权限不能严于父类


抽象类


Pet pet = new Pet ("贝贝",20,40);

pet.print();

实例化Pet没有意义

public abstract class Pet {     
}

以下代码的问题:

public abstract class Pet { 
       public void print() {
               //…
       }
}

每个子类的实现不同,抽象类中无法给出合理的print方法的实现

abstract也可用于方法——抽象方法

  • 抽象方法没有方法体
  • 抽象方法必须在抽象类里
  • 抽象方法必须在子类中被实现,除非子类是抽象类

public abstract class Pet { 
       public abstract   void print() {
               //…
       }
}

修改Pet类为抽象类

修改Pet类的print()方法为抽象方法

输出Dog信息


image.png

final用法

Penguin类不希望再被其他类继承?

使用final类

public final class Penguin extends Pet { 
      //…
}

方法不希望被重写?

使用final方法

public final void print () { 
      //…
}

属性值不希望被修改?

使用常量

public class Penguin {
       final String  home ="南极";// 居住地
       public void setHome(String name){
       this.home=home; //错误,不可再赋值
    }
}

练习

汽车租赁公司出租多种车辆,车型和租金情况如下表所示。编写程序实现租赁价格的计算。


image.png


具体要求:车辆分为轿车和客车两大类,它们都继承自抽象类MotoVehicle,并实现其抽象方法calRent()。请根据下面给出的类图分别创建三个类,并在测试类TestRent中实现车辆的租赁。租赁过程如图所示。


image.png


image.png


MotoVehicle类

public abstract class MotoVehicle {
    private String no;//车牌号
    private String brand;//品牌
    abstract int calRent(int days);//计算租金
    public MotoVehicle()
    {
    }
    public MotoVehicle(String no, String brand)
    {
        this.no = no;
        this.brand = brand;
    }
    public String getNo() {
        return no;
    }
    public void setNo(String no) {
        this.no = no;
    }
    public String getBrand() {
        return brand;
    }
    public void setBrand(String brand) {
        this.brand = brand;
    }
}

Car

public class Car extends MotoVehicle{
    public Car(String no, String brand)
    {
        super(no,brand);
    }
    @Override
    int calRent(int days) {
        if(getBrand().equals("宝马"))
        {
            return 500 * days;
        }
        else
        {
            return 600 * days;
        }
    }
}

Bus

public class Bus extends MotoVehicle {
    private int seatCount;
    public int getSeatCount() {
        return seatCount;
    }
    public void setSeatCount(int seatCount) {
        this.seatCount = seatCount;
    }
    public Bus(String no, String brand, int seatCount)
    {
        super(no,brand);
        this.seatCount = seatCount;
    }
    @Override
    int calRent(int days) {
        return 0;
    }
}

测试类

public class TestRent {
    public static void main(String[] args) {
        System.out.println("欢迎");
        System.out.println("请输入天数");
        Scanner scanner =  new Scanner(System.in);
        int days = scanner.nextInt();
        System.out.println("请输入汽车类型1.轿车,2.客车");
        int type = scanner.nextInt();
        if(type == 1)
        {
            System.out.println("输入品牌");
            String brand = scanner.next();
            Car car = new Car("辽N12345",brand);
            int money = car.calRent(days);
            System.out.println("租金为"+money);
        }
        else
        {
//            Bus bus = new Bus();
        }
    }
}


目录
相关文章
|
5月前
|
安全 程序员 编译器
C++之继承
C++之继承
|
5月前
|
安全 编译器 程序员
【C++】—— 继承
【C++】—— 继承
|
3月前
|
编译器 C++
【C++】详解C++的继承
【C++】详解C++的继承
|
5月前
|
编译器 C++
【继承】初步了解C++继承
【继承】初步了解C++继承
|
12月前
|
存储 编译器 程序员
c++【继承】
C++ 继承,包括继承的概念和用法,菱形继承的产生,组合的介绍等丰富知识点,详细讲解,干货满满!
36 4
c++【继承】
|
5月前
|
存储 安全 编译器
|
5月前
|
存储 设计模式 Java
C++【继承】
C++【继承】
64 0
|
11月前
|
安全 Java 编译器
C++中的继承(超详细)
C++中的继承(超详细)
55 0
|
12月前
|
Java C++