Java中10种覆盖方法的方式

来源:互联网 发布:淘宝旗舰店查真假 编辑:程序博客网 时间:2024/05/21 22:38

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 B :Function fb Runing......
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(); }


0 0
原创粉丝点击