学习java的第十三课——方法重写与多态

本文详细讲解了Java中的方法重写和多态概念。首先介绍了方法重写的基本原则,包括如何使用super关键字调用父类方法,并讨论了构造方法在继承中的应用。接着,探讨了Object类的equals()方法和重写它的必要性。然后,解释了多态的概念及其在代码优化中的作用。最后,提到了抽象类和抽象方法,以及在继承和向上向下转型中的使用。
学习java的第十三课——方法重写与多态
初识方法重写

方法的重写或方法的覆盖

​ 子类根据需求对从父类继承的方法进行重新编写

​ 重写时,可以用super.方法的方式来保留父类的方法

​ 构造方法不能被重写

方法重写的条件

​ 两个类:有继承关系的父子类

​ 同名方法(疑问:是否必须返回值 访问修饰符等都相同??)

super的介绍与使用

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-mlL8Xt4q-1638170281763)(C:\Users\98710\AppData\Roaming\Typora\typora-user-images\image-20211115184919618.png)]

调用父类的非private方法

super可以调用父类的非private属性

System.out.println(super.name);

可以访问父类的构造方法

super关键字来访问父类成员

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

​ super只能出现在子类的方法和构造方法中

​ super调用构造方法是,只能是第一句

​ super不能访问父类的private成员

super可以使被屏蔽的成员可见

在这里插入图片描述

不能

Java只支持单根继承

一个类只能有一个直接父类

一个类可以有多个间接父类

在这里插入图片描述

继承下的构造方法

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

​ 子类构造方法没有通过super显式调用父类的有参构造方法也没通过this显式调用自身其他构造方法

​ 系统默认调用父类的构造方法

​ 子类构造方法通过super显式调用父类的有参构造方法

​ 执行父类相应的构造方法,而不是父类无参构造方法

​ 子类构造方法通过this显式调用自身的其他构造方法,在相应构造方法中应用以上两条规则
在这里插入图片描述

Bus是子类,Car是父类

答案:载客量是4人,载客量是20人

继承条件下的构造方法执行

​ 控制台输出语句

​ 断点跟踪了解过程

​ 助于深入理解继承以及super使用

方法重写深入

在这里插入图片描述

重写了Pet中的print方法

方法重写的规则

方法名相同

参数列表相同

返回值类型相同或者是其子类

访问权限不能严于父类

​ 父类的静态方法不能被子类覆盖为非静态的方法,父类的非静态方法不能被子类覆盖为静态方法

​ 子类可以定义与父类同名的静态方法,以便在子类中隐藏父类的静态方法(静态方法中无法使用***super***)

​ 父类的私有方法不能被子类覆盖

​ 不能抛出比父类方法更多的异常

在这里插入图片描述

Object类

Object类是所有类的直接或间接父类

public class Pet extends Object{

//代码块
}

Object : 对象

Object类的equals()方法

比较两个对象是否为同一个对象,是则返回true

操作符 ==

​ 简单数据类型,直接比较值.比如1==2

​ 引用类型,比较两者是否为同一对象

 public static void main(String[] args) {
        Stundent s1 = new Stundent();
        Stundent s2 = new Stundent();
        //判断两个对象是否为同一个对象
        System.out.println(s1 == s2);
        System.out.println(s1.equals(s2));
    }

结果为false false

为什么结果为false???

因为是s1是在栈中,new一个对象是在堆内存中,s2也是,所以都是false

   Stundent s1 = new Stundent();
        Stundent s2=s1;//将s1这个对象传给s2传的是地址
        //判断两个对象是否为同一个对象
        System.out.println(s1 == s2);
        System.out.println(s1.equals(s2));


    }

结果是true true

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-CTFCY0Jx-1638170281771)(C:\Users\98710\AppData\Roaming\Typora\typora-user-images\image-20211115200124850.png)]

他两个是同一个对象所以结果为true

Object类中的equals()方法和==没有区别

Object类中的equals()方法中和==作用相同(引用类型),判断是否为统一对象

重写Obect类的方法
 Stundent s1 = new Stundent(1,"张三",18,55);
        Stundent s2 = new Stundent(1,"张三",18,55);
        //判断两个对象是否为同一个对象
        System.out.println(s1 == s2);
        System.out.println(s1.equals(s2));

以上判断为不是同一个对象

但是在现实生活中同学号姓名年龄体重的是同一个对象所以要重写Object类中的equals方法

快捷键ctrl+o选择equals

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-MLzod0Ak-1638170281771)(C:\Users\98710\AppData\Roaming\Typora\typora-user-images\image-20211115200919958.png)]

instanceof运算符 作用:判断某一个对象是否属于一个类型 判断a是否为b类型👇

传入的数据类型a instanceof 要判断的数据类型b

重写Object

  public boolean equals(Object obj) {
        //如果当前对象this  和传入的obj通过==(内存地址相同)是true那么两者为同一对象,肯定是true
        if (this == obj){
            return true;
        }
        //如果传入的obj对象不是Stundent类型(不是我们要比的数据类型),那么一定是false
        //instanceof运算符 作用:判断某一个对象是否属于一个类型

        if(!(obj instanceof Stundent)){//不是学生对象直接false
        return false;
        }//排除上面情况后代表传入的数据一定是学生类型
        //需要把他强转为学生类型
        //因为传入的obj是继承于Object的类型
        //强转后赋值给s1
        Stundent s1 = (Stundent) obj;
        //直接比较s和当前对象this的id以及name即可
        if(this.id==s1.id && this.name.equals(s1.name)){
            return true;
        }else{
            return false;
        }

    }

重写了equals()方法,把equals()方法的判断变为两个值之间的判断

toString()返回一个字符串的信息

在这里插入图片描述

//重写toString():对象信息不要返回内存地址的信息而是要返回学生的姓名
    @Override
    public String toString() {
        return this.name;
    }

直接改返回值为this.name;

当我们直接输出对象时,直接输出toString

//作用相同直接输出toString
System.out.println(s1.toString());
System.out.println(s1);
//String类不仅重写了equals方法,还重写了toString()方法
String s = "jkl";
System.out.println(s.toString());
System.out.println(s);

Object类的方法可以根据需求自行重写

equals()

toString():输出对象信息时,默认调用

Java.lang.String重写了equals()方法,把equals()方法的判断变为了判断其值,也重写了toString()方法

instanceof用于判断一个引用类型所引用的对象是否为一个类的实例

什么是多态

在这里插入图片描述

添加XX类,继承Pet类

修改主人类,添加给XXX看病的方法

这样会导致频繁修改代码,代码可拓展性,可维护性差,所以要使用多态优化
在这里插入图片描述

多态:同一个引用类型,使用不同的实例而执行不同操作

狗狗类

public class Dog extends Pet{
	public void toHospital(){
		this.setHealth(60);
		System.out.print("吃药,打针")
	
	}


}

企鹅类

public class Penguin extends Pet{
public void toHospital(){
		this.setHealth(60);
		System.out.print("吃药,疗养")
}

主人类

public class Master{
public void cure(Pet pet){//传入一个宠物类,如果他的健康值低于50,则给他医治
if(pet.getHealth()<50){
	pet.toHospital();
}

}

}

测试方法

Pet pet = new Dog();
Master m1 = new Master();
master.cure(pet);//父类引用子类对象

继承和方法重写是实现多态的基础
在这里插入图片描述

抽象类和抽象方法

抽象方法

​ 抽象方法没有方法体

​ 抽象方法必须在抽象类中

​ 抽象方法必须在子类中实现,除非子类是抽象类

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-2uOIlgrT-1638170281774)(C:\Users\98710\AppData\Roaming\Typora\typora-user-images\image-20211116185518732.png)]

向上向下转型

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gymuotm2-1638170281774)(C:\Users\98710\AppData\Roaming\Typora\typora-user-images\image-20211116185815762.png)]

此时通过父类引用变量调用的方法是子类覆盖或继承父类的方法

此时通过父类引用变量无法调用子类特有的方法.

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Kl5dUX0q-1638170281775)(C:\Users\98710\AppData\Roaming\Typora\typora-user-images\image-20211116190046461.png)]

偈减少再向下转型的过程中,没有转换为真实子类类型的类型转换异常?

在Java中提供了instanceof运算符来进行类型的判断

使用instance偶分\时,对象和类型必须和instanceof后面参数所指的类在继承上有上下级关系

<父类型><引用变量名>=new <子类型>();
//自动转换
<子类型><引用变量名>=(<子类型>)<父类型的引用变量>;
//强制转换
父类作为返回值的练习

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值