第四章 方法重写与多态
1.方法重写
方法的重写或方法的覆盖(overriding)
- 子类根据需求对从父类继承的方法进行重新编写
- 重写时,可以用super.方法的方式来保留父类的方法
- 构造方法不能被重写
方法重写规则
方法名相同
参数列表相同
返回值类型相同或者是其子类
访问权限不能严于父类
父类的静态方法不能被子类覆盖为非静态方法,父类的非静态方法不能被子类覆盖为静态方法
子类可以定义与父类同名的静态方法,以便在子类中隐藏父类的静态方法
父类的私有方法不能被子类覆盖不能抛出比父类方法更多的异常
(注:静态方法中无法使用super)
方法重写vs方法重载
比较项 | 位置 | 方法名 | 参数表 | 返回值 | 访问修饰符 |
---|---|---|---|---|---|
方法重写 | 子类 | 相同 | 相同 | 相同或是其子类 | 不能比父类更严格 |
方法重载 | 同类 | 相同 | 不相同 | 无关 | 无关 |
2.Object类
Object类是所有类的父类
Object类被子类经常重写的方法:
方法 | 说明 |
---|---|
toString() | 返回当前对象本身的有关信息,按字符串对象返回 |
equals() | 比较两个对象是否是同一个对象,是则返回true |
hashCode() | 返回该对象的哈希代码值 |
getClass() | 获取当前对象所属的类信息,返回Class对象 |
//重写equals
public boolean equals(Object obj){
//将子类转换为父类(向下转型)
Dog d = (Dog)obj;
if(this.getName().equals(d.getName))
&& this.getHealth() == d.getHealth()
&& this.getLove() == d.getLove()
&& this.getType().equals(d.getType()){
return true;
}
return false;
}
Object类的equals()方法:
- 比较两个对象是否是同一个对象,是则返回true
- 操作符==
- 简单数据类型,直接比较值。如1==2
- 引用类型,比较两者是否为同一对象
注意:
(1)Object类的equals()方法与==没区别
(2)当有特殊需求,如认为属性相同即为同一对象时,需要重写equals()
(3)Java.lang.String重写了equals()方法,把equals()方法的判断变为了判断其值
3.多态
同一种事物,由于条件不同,产生的结果也不同
同一个引用类型,使用不同的实例而执行不同操作
方法重写是实现多态的基础
4.抽象方法
public abstract class Pet {
public void toHospital() {
}
}
- 抽象方法没有方法体
- 抽象方法必须在抽象类里
- 抽象方法必须在子类中被实现,除非子类是抽象类
5.向上转型
向上转型:父类的引用指向子类对象,自动进行类型转换
//测试方法
Pet pet = new Dog();
pet.setHealth(20);
Master master = new Master();
master.cure(pet);
<父类型> <引用变量名> = new <子类型>();
注意:
此时通过父类引用变量调用的方法是子类覆盖或继承父类的方法,不是父类的方法
此时通过父类引用变量无法调用子类特有的方法
6.向下转型
Dog dog=(Dog)pet;//将pet 转换为Dog类型
dog. catchingFlyDisc();//执行Dog特有的方法
<子类型> <引用变量名> = (<子类型> )<父类型的引用变量>;
注意:
在向下转型的过程中,如果没有转换为真实子类类型,会出现类型转换异常
instanceof运算符来进行类型的判断
*使用instanceof时,对象的类型必须和instanceof后面的参数所指定的类在继承上有上下级关系
7.多态的应用
-
使用父类作为方法的形参
//为宠物看病 public void cure(Pet pet) { if (pet.getHealth() < 50) pet.toHospital(); }
-
使用父类作为方法的返回值
//测试方法 Pet pet = new Dog(); pet.setHealth(20); Master master = new Master(); master.cure(pet);