1)子类方法的名称、参数签名和返回类型必须与父类方法的名称、参数签名和返回类型一致,修饰符可以相同也可以不同,但子类的访问权限不能低于父类的访问。
class Aball {
Aball() {
}
public void method() {
}
}
public class ExtendsMethod extends Aball {
ExtendsMethod() {
}
public void method() {//覆盖父类的方法
}
public int method(int i) {//重载类中的方法
i++;
return i;
}
}
2)子类方法不能缩小父类方法的访问权限。
3)子类方法不能抛出比父类方法更多的异常。子类方法抛出的异常必须和父类方法抛出的异常相同,或者子类方法抛出的异常类是父类方法抛出的异常类的子类。另外,写抛出异常catch块时,子类方法抛出的异常在父类方法抛出异常的前面。
4)方法覆盖只存在于子类和父类(包括直接父类和间接父类)之间。在同一个类中方法只能被重载,不能被覆盖。
5)父类的静态方法不能被子类覆盖为非静态方法。
6)子类可以定义与父类的静态方法同名的静态方法,以便在子类中隐藏父类的静态方法。(静态方法只能隐藏,不能覆盖)
子类隐藏父类的静态方法和子类覆盖父类的实例方法区别:运行时,JVM把静态方法和所属的类绑定,而把实例方法和所属的实例绑定。
class Aball {
Aball() {
}
void method() {
System.out.println("method of Aball");
}
static void staticmethod() {
System.out.println("staticmethod of Aball");
}
}
public class ExtendsMethod extends Aball {
ExtendsMethod() {
}
void method() {// 覆盖父类的方法
System.out.println("method of ExtendsMethod");
}
static void staticmethod() {// 重载类中的方法
System.out.println("staticmethod of ExtendsMethod");
}
public static void main(String[] args) {
Aball sub1 = new ExtendsMethod();
ExtendsMethod sub2 = new ExtendsMethod();
sub1.method();
sub1.staticmethod();
sub2.method();
sub2.staticmethod();
}
}
结果为:
method of ExtendsMethod
staticmethod of Aball
method of ExtendsMethod
staticmethod of ExtendsMethod
引用变量sub1和sub2都引用ExtendsMethod类的实例,执行sub1.method()和sub2.method()时,都调用ExtendsMethod实例的method()方法,此时父类Aball的实例方法method()被子类覆盖。
引用变量sub1被声明为Aball类型,执行sub1.staticmethod()时,调用Aball类的staticmethod()方法,说明父类的静态方法不能被子类覆盖。
引用变量sub2被声明为ExtendsMethod类型,执行sub2.staticmethod()时,调用ExtendsMethod类的staticmethod()方法,说明父类的静态方法被子类的静态方法隐藏了。
7)父类的非静态方法不能被子类覆盖为静态方法。
8)父类的私有方法不能被子类覆盖。
class Aball {
Aball() {
}
private String method() {
return "Aball";
}
void print() {
System.out.println(method());
}
}
public class ExtendsMethod extends Aball {
ExtendsMethod() {
}
public String method() {// 覆盖父类的方法
return "ExtendsMethod";
}
public static void main(String[] args) {
Aball sub1 = new Aball();
ExtendsMethod sub2 = new ExtendsMethod();
sub1.print();
sub2.print();
}
}
结果为:
Aball
Aball
执行sub2.print()方法时,因为print()方法在Aball类中定义,因此Aball方法会调用在Aball类中定义的private类型的method()方法。
把Aball类的method()方法改为public类型,期他代码不变,
则运行结果为
Aball
ExtendsMethod
原因是由于ExtendsMethod中的method()方法覆盖了Aball类中的method()方法,执行sub2.print()时,JVM会调用当前ExtendsMethod实例的method()方法。
看下面的例子:
public class Polymorphism{
public static void main(String[] args) {
A b = new B();
b.fb();
}
}
class A {
public A(){
}
public void fa() {
System.out.println("CLASS A :Function fa Runing......");
}
public void fb() {
System.out.println("CLASS A :Function fb Runing......");
fa();
System.out.println("CLASS A :Function fb Stop......");
}
}
class B extends A {
public B(){
}
public void fa() {
System.out.println("CLASS B :Function fa Runing......");
}
public void fb() {
System.out.println("CLASS B :Function fb Runing......");
super.fb();
System.out.println("CLASS B :Function fb Stop......");
}
}
下面是它的运行结果:
CLASS A :Function fb Runing......
CLASS B :Function fa Runing......
CLASS A :Function fb Stop......
CLASS B :Function fb Stop......
1、运行main函数,创建B对象,调用B的方法fb,于是打印出"CLASS B :Function fb Runing......",都在情理之中。
2、执行super.fb(),调用父类A的方法fb,首先打印出"CLASS A :Function fb Runing......",预料之中
3、执行方法fa(),打印出"CLASS B :Function fa Runing......",呃?奇怪了,为什么不是执行A的方法fa(),而是子类B中的fa()呢? 当前被执行的是类A的方法,那么虚拟机找到的应该是A类的Method Table,找到的应该是A类的方法fa()啊? 难解~
4、打印"CLASS A :Function fb Stop......",返回
5、打印 " CLASS A :Function fb Stop.....",返回,程序结束。
现在问题清楚了,就是虚拟机在执行类A方法的时候查找的Method Table竟然是子类B的。为什么呢?其实,只要我们清楚java方法调用的方 式,这个问题就迎刃而解了。在Java虚拟机中,每启动一个新的线程,虚拟机都会为它分配一个Java栈,而每当线程调用一个java方法时,虚拟机就会 在该线程的java栈中压入一个新帧,用以存储参数,局部变量等数据。我们将这个正在被执行的方法称为该线程的当前方法,其相应的栈帧为当前帧。
好了,当我们调用一个方法时,我们需要往当前帧中压入哪些参数呢?简单,方法的参数列表中不是都说得清清楚楚的吗?嗯,对于C语言来说,这个说法 是正确的,但是对于诸如C++,Java,Python等面向对象语言来说,却是不对的。大家还记得那个"this"指针吗?!不错, 在Java中,所有的实例方法(Instance Method)调用的时候都会把当前对象压入当前帧中,Java虚拟机正是通过这个参数来决定当前所使用的类(通过判断该对象的类型) 。
在上面的例子中,main中调用b.fb()时,压入的当前对象自然是B类对象,我们记为b。在B的fb()中调用super.fb()时,压入 的就是刚刚压入的对象,也就是b了。同样,在A的fb中调用fa()时,压入的也是b。因此,在使用 invokevirtual指令调用fa()时,找 的就是B的方法表(当前对象b的类型为B),也就执行了类B的fa了。
这种现象在构造函数中特别常见,因为构造函数中会隐含使用调用父类的构造函数的,如果在父类的构造函数中调用了实例方法(如 A的fa),而在子 类中又覆盖了这个实例方法(如 B的fa),那么得到的结果往往不是我们所要的。因此,我们最好不要在构造函数中使用多态方法,不然,Debug会很痛苦 的:)
9)父类的抽象方法可以被子类通过两种途径覆盖:一是子类实现父类的抽象方法;二是子类重新声明父类的抽象方法。
abstract class Aball {
Aball() {
}
abstract void method();
abstract void print();
}
public abstract class ExtendsMethod extends Aball {
ExtendsMethod() {
}
public void method() {// 实现method()方法,并扩大访问权限
}
// 重新声明print()方法,并扩大访问权限,但不实现
public abstract void print();
}
10 )父类的非抽象方法可以被覆盖为抽象方法。
abstract class Aball {
Aball() {
}
void method() {
}
void print() {
}
}
public abstract class ExtendsMethod extends Aball {
ExtendsMethod() {
}
public void method() {// 覆盖父类的method()方法
}
// 覆盖父类的print()方法
public abstract void print();
}