在面向对象方面,方法的重载(overload)和覆盖(override)是常考的面试点,一般的问题是,它们有什么差别?下面我们来通过DifferenceDemo.java来看重载和覆盖的差别。
1 class Base{
2 public void print(){}
3 //是重载,因为参数不同
4 public void print(int row){}
5 //是重载,因为参数类型不同
6 public void print(String type){}
7 }
8 class Child extends Base{
9 //是覆盖,覆盖了父类的同名无参方法
10 public void print(){}
11 //出错,因为父类已经有个带int参的同名方法
12 //public String print(int row){}
13 //是覆盖,覆盖了父类的带String参的同名方法
14 public void print(String type){}
15 //是重载,因为参数个数不同
16 public void print(int row, String type){}
17 }
18
19 public class DifferenceDemo {
20 public static void main(String[] args) { }
21 }
在Base类里,我们定义了3个print方法,它们的参数类型或个数不同,这属于重载,在Child类的第16行里,我们定义了带2个参数的方法,这个也是重载。
在第8行的Child类里,我们继承了Base类,第10行的无参方法,和Base类里第2的方法完全一致(方法名,参数和返回值完全一致),这个是覆盖,子类里能用完全一致的方法覆盖掉父类方法。同样的,在子类里第14行定义的带一个String类型的方法也能覆盖掉父类里第6行的同名同参同返回值的方法。
不过请大家注意第12行的方法,和父类第4行定义的带一个int参的方法相比,它的返回值不同,父类是返回void,而这里是String,这种不属于覆盖,也不属于重载,是语法错误,对此Java编译器会报“定义了重复方法”这个错误。
判断重载和覆盖有个比较通俗的依据,我们可以把子类里的方法移动到父类里。
第一,如果方法名,参数和返回类型都一致,那么我们可以说是子类的方法覆盖了父类的。
第二,如果方法同名,参数个数或类型不同,这时无需看返回类型,这种情况属于重载。
第三,当把子类里的方法移动到父类后,两个同名方法参数完全一致,但返回类型不同,这种是语法错误,Java编译器会报“定义了重复方法”这个错误。
请注意,如果你能讲清楚在覆盖父类方法时的注意点,面试官一定会认为你很资深。
第一,子类方法不能缩小父类方法的访问权限,比如我们在上述第5行覆盖父类的方法时,修饰符可以是protected,也可以是更大的public,但不能是缩小的private。在父类里定义的private方法子类里也看不到,所以也不存在缩小访问权限的问题,如果我们在父类里定义protected或public或默认的方法,总是希望子类能用到或重写,但如果我们一旦缩小范围,比如把第5行的protected缩小成private,那么ChinsePerson的子类就看不到爷爷类(也就是Person)类的speak方法了,就会造成“父类方法失传”这个问题,所以Java干脆从语法上规定了这点以杜绝这类问题。
第二,子类方法不能抛出比父类方法更多的异常。现在请大家先记得这点,原因等后面我们讲异常处理时再分析。
?更多java面向对象面试题的讲解分析:
Java面向对象面试题讲解:类和实例的差别
Java面向对象面试题分析,你知道哪些Java访问修饰符
Java面向对象面试题分析:静态变量和静态方法
Java面向对象面试题分析:说出抽象类和接口的差别
Java面向对象面试题分析:说出final关键字的用法
Java面向对象面试题解析:说下方法的覆盖和重载
|