重写
Override 翻译为 覆写 或者 重写 都可以,它指在子类在继承父类时,如果父类已经存在一个方法 ,子类可以继续沿用该方法,但必须保证在子类中该方法的 方法签名 与父类的 方法签名 必须一致,且必须要保证子类中该方法的 返回值类型 要么是父类该方法的 返回值类型,要么是父类该方法的 返回值类型 的子类型。在Java中 方法签名 相同可以方法的简单地理解为如果两个方法的 方法名、参数个数 和 参数类型顺序 都相同,那么我们就称这两个方法的 方法签名 相同。
常用的重写案例
interface Operation{
public int operate(int a, int b);
}
class Addition implements Operation{
@Override
public int operate(int a, int b){
// 方法签名与返回值类型均与父类相同
return a + b;
}
}
符合规则的重写案例(返回类型不一致)
/**
奖励
*/
class Award{
}
/**
现金
*/
class Cash extends Award{
}
class Task {
}
/*-----------------------------------------------------------------------*/
/**
比赛
*/
class Competition{
public Award compelete(Task task){
// 赢得比赛,获得奖励
return new Award();
}
}
/**
买彩票
*/
class Lottery extends Competition{
@Override
public Cash compelete(Task task){
// 中奖,获得现金
return new Cash();
}
}
重载
重载 在编程语言中对应的英语术语为 Overload 。它指在同一个类中,可以定义同名的方法,但是这些方法的 参数个数 或者 参数类型顺序 必须不一致,返回值类型 是否相同并不做要求。
重载的方法在编译过程中即可完成识别。具体到每一个方法调用,Java 编译器会根据所传入参数的声明类型(注意与实际类型区分)来选取重载方法。选取的过程共分为三个阶段:
- 在不考虑对基本类型自动装拆箱,以及可变长参数的情况下选取重载方法
- 如果在第 1 个阶段中没有找到适配的方法,那么在允许自动装拆箱,但不允许可变长参数的情况下选取重载方法;
- 如果在第 2 个阶段中没有找到适配的方法,那么在允许自动装拆箱以及可变长参数的情况下选取重载方法。
如果 Java 编译器在同一个阶段中找到了多个适配的方法,那么它会在其中选择一个最为贴切的,而决定贴切程度的一个关键就是形式参数类型的继承关系:在继承树中越靠下越贴切。
除了同一个类中的方法,重载也可以作用于这个类所继承而来的方法。也就是说,如果子类定义了与父类中非私有方法同名的方法,而且这两个方法的参数类型不同,那么在子类中,这两个方法同样构成了重载。
一个简单的重载案例
class Addition {
public int operate(int a, int b){
return a + b;
}
public long operate(long a, long b){
// 与第一个函数的 参数类型 不一致
return a + b;
}
public int operate(int a, int b, int c){
// 与第一个函数的 参数个数 不一致
return a + b + c;
}
public int operate(int a, char b){
// 与第一个和第二个函数的 参数类型 不一致
return a + b;
}
public int operate(char a, int b){
// 与第四个函数的 参数类型顺序 不一致
return a + b;
}
}
深入 JVM
Java 虚拟机识别方法的关键在于类名、方法名以及方法描述符(method descriptor)。至于方法描述符,它是由方法的参数类型以及返回类型所构成。在同一个类中,如果同时出现多个名字相同且描述符也相同的方法,那么 Java 虚拟机会在类的验证阶段报错。
可见,Java 虚拟机与 Java 语言不同,它并不限制名字与参数类型相同,但返回类型不同的方法出现在同一个类中,对于调用这些方法的字节码来说,由于字节码所附带的方法描述符包含了返回类型,因此 Java 虚拟机能够准确地识别目标方法。
Java 虚拟机中关于方法重写的判定同样基于方法描述符。也就是说,如果子类定义了与父类中非私有、非静态方法同名的方法,那么只有当这两个方法的参数类型以及返回类型一致,Java 虚拟机才会判定为重写。 对于 Java 语言中重写而 Java 虚拟机中非重写的情况,编译器会通过生成桥接方法来实现 Java 中的重写语义。