Java中10种覆盖方法的方式
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 ExtendsMethodstaticmethod of Aballmethod of ExtendsMethodstaticmethod 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(); }} 结果为:AballAball 执行sub2.print()方法时,因为print()方法在Aball类中定义,因此Aball方法会调用在Aball类中定义的private类型的method()方法。 把Aball类的method()方法改为public类型,期他代码不变,则运行结果为AballExtendsMethod 原因是由于ExtendsMethod中的method()方法覆盖了Aball类中的method()方法,执行sub2.print()时,JVM会调用当前ExtendsMethod实例的method()方法。 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(); } 1、方法继承:利用extends关键字一个方法继承另一个方法,而且只能直接继承一个类。
当Sub类和Base类在同一个包时Sub类继承Base类中的public/protected/默认级别的变量个方法
在不同包时继承public/protected级别的变量和方法。
2、方法重载:如果有两个方法的方法名相同,但参数不一致,哪么可以说一个方法是另一个方法的重载。
方法名相同
方法的参数类型,个数顺序至少有一项不同
方法的返回类型可以不相同
方法的修饰符可以不相同
main方法也可以被重载
3、方法覆盖:如果在子类中定义一个方法,其名称、返回类型及参数签名正好与父类中某个方法的名称、返回类型及参数签名相匹配,那么可以说,子类的方法覆盖了父类的方法。
子类的方法名称返回类型及参数签名 必须与父类的一致
子类方法不能缩小父类方法的访问权限
子类方法不能抛出比父类方法更多的异常
方法覆盖只存在于子类和父类之间,同一个类中只能重载
父类的静态方法不能被子类覆盖为非静态方法
子类可以定义于父类的静态方法同名的静态方法,以便在子类中隐藏父类的静态方法(满足覆盖约束),
而且Java虚拟机把静态方法和所属的类绑定,而把实例方法和所属的实例绑定。
父类的非静态方法不能被子类覆盖为静态方法
父类的私有方法不能被子类覆盖
父类的抽象方法可以被子类通过两种途径覆盖(即实现和覆盖)(P169)
父类的非抽象方法可以被覆盖为抽象方法
4、Super关键字:super和this关键字都可以用来覆盖Java语言的默认作用域,使被屏蔽的方法或变量变为可见(三种情况下的不可见P171)。
父类的成员变量和方法为private使用super访问编译出错
在类的构造方法种,通过super语句调用这个类的父类的构造方法
在子类种访问父类的被屏蔽的方法和属性
只能在构造方法或实例方法内使用super关键字,而在静态方法和静态代码块内不能使用super
5、多态:
对于一个引用类型的变量,Java编译器按照它的声明的类型来处理
对于一个引用类型的变量,运行时Java虚拟机按照它的实际引用的对象来处理
运行时环境中,通过引用类型变量来访问所引用对象的方法和属性时,Java虚拟机采用以下绑定规则
1)实例方法与引用变量实际引用的对象的方法绑定,属于动态绑定
2)静态方法与引用变量所声明的类型的方法绑定,属于静态绑定
3)成员变量(包括静态和实例变量)与引用变量所声明的类型的成员变量绑定,属于静态绑定
6、继承的利弊和使用原则:
集成数的层次不可太多
集成数的上层为抽象层
(1)定义了下层子类都用友的相同属性和方法,并且尽可能默认实现,从而提高重用性
(2)代表系统的接口,描述系统所能提供的服务
继承关系最大的弱点:打破封装
精心设计专门用于被继承的类
(1)对这些类必须提供良好的文档说明
(2)尽可能的封装父类的实现细节,把代表时间细节的属性和方法定义为private类型
(3)如果某些实现细节必须被子类访问,定义为protected类型
(4)把不允许子类覆盖的方法定义为final类型
(5)父类的构造方法不允许调用可被子类覆盖的方法
(6)如果某些类不是专门为了继承而设计,那么随意继承它是不安全的