解析Java继承中方法的覆盖和重载

时间:2021-12-05 23:59:03

方法的覆盖

在类继承中,子类可以修改从父类继承来的方法,也就是说子类能创建一个与父类方法有不同功能的方法,但具有相同的名称、返回值类型、参数列表。

  如果在新类中定义一个方法,其名称、返回值类型和参数列表正好与父类中的相同,那么,新方法被称做覆盖旧方法。

  参数列表又叫参数签名,包括参数的类型、参数的个数和参数的顺序,只要有一个不同就叫做参数列表不同。

  被覆盖的方法在子类中只能通过super调用。

  注意:覆盖不会删除父类中的方法,而是对子类的实例隐藏,暂时不使用。

  请看下面的例子:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
public class Demo{
public static void main(String[] args) {
Dog myDog = new Dog("行行圈");
myDog.say(); // 子类的实例调用子类中的方法
Animal myAnmial = new Animal("行行圈在线");
myAnmial.say(); // 父类的实例调用父类中的方法
}
}
class Animal{
String name;
public Animal(String name){
this.name = name;
}
public void say(){
System.out.println("我是一只小动物,我的名字叫" + name + ",我会发出叫声");
}
}
class Dog extends Animal{
// 构造方法不能被继承,通过super()调用
public Dog(String name){
super(name);
}
// 覆盖say() 方法
public void say(){
System.out.println("我是一只小狗,我的名字叫" + name + ",我会发出汪汪的叫声");
}
}

  运行结果:

 

复制代码 代码如下:

我是一只小狗,我的名字叫行行圈,我会发出汪汪的叫声
我是一只小动物,我的名字叫行行圈在线,我会发出叫声

 

方法覆盖的原则:

  覆盖方法的返回类型、方法名称、参数列表必须与原方法的相同。

  覆盖方法不能比原方法访问性差(即访问权限不允许缩小)。

  覆盖方法不能比原方法抛出更多的异常。

  被覆盖的方法不能是final类型,因为final修饰的方法是无法覆盖的。

  被覆盖的方法不能为private,否则在其子类中只是新定义了一个方法,并没有对其进行覆盖。

  被覆盖的方法不能为static。如果父类中的方法为静态的,而子类中的方法不是静态的,但是两个方法除了这一点外其他都满足覆盖条件,那么会发生编译错误;反之亦然。即使父类和子类中的方法都是静态的,并且满足覆盖条件,但是仍然不会发生覆盖,因为静态方法是在编译的时候把静态方法和类的引用类型进行匹配。

方法的重载:

  前面已经对Java方法重载进行了说明,这里再强调一下,Java父类和子类中的方法都会参与重载,例如,父类中有一个方法是 func(){ ... },子类中有一个方法是 func(int i){ ... },就构成了方法的重载。

覆盖和重载的不同:

  方法覆盖要求参数列表必须一致,而方法重载要求参数列表必须不一致。

  方法覆盖要求返回类型必须一致,方法重载对此没有要求。

  方法覆盖只能用于子类覆盖父类的方法,方法重载用于同一个类中的所有方法(包括从父类中继承而来的方法)。

  方法覆盖对方法的访问权限和抛出的异常有特殊的要求,而方法重载在这方面没有任何限制。

  父类的一个方法只能被子类覆盖一次,而一个方法可以在所有的类中可以被重载多次。