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 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(); }
- Java中10种覆盖方法的方式
- Java中10种覆盖方法的方式
- Java中方法的覆盖
- java 中equals方法的覆盖
- Java中方法覆盖的注意事项
- java中方法覆盖的相关
- java方法的覆盖
- java 方法的覆盖
- Java的方法覆盖
- java 中方法重载和方法覆盖的却别
- java中属性的隐藏和方法的覆盖区别
- java类的继承中方法的覆盖与重载
- Java 中方法的重写,重载与覆盖的关系
- 弄清Java中方法覆盖时的规则
- C++和Java继承关系中方法覆盖的不同
- java中方法的覆盖与覆写
- Java继承中方法的覆盖和重载
- Java继承中方法的覆盖和重载
- 白血病男子帮上千病友购仿制药涉嫌“销售假药罪”被诉
- TCP/IP、Http、Socket的区别
- 使用JDOM解析XML文件
- java中获取日期的差值
- Android本地广播机制
- Java中10种覆盖方法的方式
- git clone 插件 + Git远程操作详解
- ping 192.168.1.1-t命令有什么用?
- ping命令的处理过程
- java学习之dom4j 对 xml的读取
- Memcached分析特点
- [转载]关于LWIP协议栈连续多次tcp_write后失败的解决过程
- iOS 之发送邮件的常用方法
- CSS2中display:table属性的用法详解