覆盖方法约束:
1、子类方法的名称、参数签名和返回类型必须与父类方法的名称、参数签名和返回类型一致。
例如以下代码将导致错误:
public class Base{
public void method(){}
}
public class Sub extends Base{
public int method(){ //编译错误,返回类型不一致
return 0;
}
}
Java编译器首先判断Sub类的method()方法与Base类的method()方法的参数签名,由于两者一致,因此Java编译器认为Sub类的method()方法试图覆盖父类的方法,既然如此,Sub类的method()方法就必须和被覆盖的方法具有相同的返回类型。
以下代码中子类覆盖了父类的一个方法,然后又定义了一个重载方法,这是合法的。
public class Base{
public void method(){}
}
public class Sub extends Base{
public void method(){} //覆盖父类Base类的method()方法
public int method(int a){ //重载method()方法
return a;
}
}
2、子类方法不能缩小父类方法的访问权限。
例如以下代码中子类的method()方法是私有的,父类的method()方法是公共的,子类缩小了父类方法的访问权限,这是无效的覆盖,将导致编译错误。
public class Base{
public void method(){}
}
public class Sub extends Base{
private void method(){} //编译错误,子类缩小了父类方法的访问权限
}
为什么子类方法不允许缩小父类方法的访问权限呢?这是因为假如没有这个限制,将会与Java语言的多态机制发生冲突。例如以下代码:
Base base = new Sub(); //base变量被定义为Base类型,但引用Sub类的实例
base.method();
Java编译器会认为以上为合法的代码。但在运行时,根据动态绑定的规则,Java虚拟机会调用base变量所引用的Sub实例的method()方法,如果这个方法为private类型,Java虚拟机就无法访问它。所以为了避免这样的矛盾,Java语言不允许子类方法缩小父类中被覆盖方法的访问权限。
3、子类方法不能抛出比父类方法更多的异常。子类方法抛出的异常必须和父类方法抛出的异常相同,或者子类抛出的异常类是父类方法抛出的异常类的子类。
例如,假设异常类ExceptionSub1和ExceptionSub2是ExceptionBase类的子类,则一下代码是合法的:
public class Base{
void method() throws ExceptionBase{}
}
public class Sub1 extends Base{
void method() throws ExceptionSub1{}
}
public class Sub2 extends Base{
void method() throws ExceptionSub2{}
}
public class Sub3 extends Base{
void method() throws ExceptionBase{}
}
以下代码不合法:
public class Base{
void method() throws ExceptionSub1{}
}
public class Sub1 extends Base{
void method() throws ExceptionBase{}
}
public class Sub2 extends Base{
void method() throws ExceptionSub1, ExceptionSub2{}
}
为什么子类方法不允许抛出比父类更多的异常呢?这个原因与子类和父类的访问权限原因类似。假如没有这个限制,将会与Java语言的多态机制发生冲突。例如以下代码:
Base base = new Sub2(); //Base类型变量base引用Sub2类的实例
try{
base.method();
}catch(ExceptionSub1 e){ //仅仅捕获父类的ExceptionSub1异常
}
在运行时,根据动态绑定规则,Java虚拟机会调用base变量所引用的Sub2实例的method()方法,假如Sub2实例的method()方法抛出ExceptionSub2异常,由于该异常没有捕获,将导致程序异常终止。
4、方法覆盖只存在于子类和父类(包括直接父类和间接父类)之间 。在同一个类中方法只能被重载,不能被覆盖。
5、父类的静态方法不能被子类覆盖为非静态方法。
例如以下代码将导致编译错误:
public class Base{
public static void method(){}
}
public class Sub extends Base{
public void method(){} //编译出错
}
6、子类可以定义与父类静态方法同名的静态方法,以便在子类中隐藏父类的静态方法。在编译时,子类定义的静态方法也必须满足类似的约束:方法的参数签名一致,返回类型一致,不能缩小父类方法的访问权限,不能抛出比父类方法更多的异常。
例如以下代码是合法的:
public class Base{
static int method(int a) throws BaseException{ return 0; }
}
public class Sub extends Base{
public static int method(int a) throws SubException{ return 0; }
}
子类隐藏父类的静态方法和子类覆盖父类的实例方法有区别,这两者的区别在于:运行时,Java虚拟机把静态方法和所属类绑定,而把实例方法与所属实例绑定。下面举例来解释这一区别,在例子中,Base类和他的子类Sub类中都定义了实例方法method()和静态方法staticMethod()。
class Base{
void method(){ //父类实例方法
System.out.println("method of Base");
}
static void staticMethod(){ //父类静态方法
System.out.println("static method of Base");
}
}
public class Sub extends Base{
void method(){ //覆盖父类的实例方法method
System.out.println("method of Sub");
}
static void staticMethod(){ //隐藏父类的静态方法staticMethod
System.out.println("static method of Sub");
}
//测试方法
public static void main(String[] args){
Base sub1 = new Sub(); //Base类型变量sub1引用Sub类实例
sub1.method(); //打印 method of Sub
sub1.staticMethod(); //打印static method of Base
Sub sub2 = new Sub(); //Sub类型变量sub2引用其本身的实例
sub2.method(); //打印 method of Sub
sub2.staticMethod(); //打印static method of Sub
}
}
引用变量sub1和sub2都引用Sub类的实例,Java虚拟机在执行sub1.method()和sub2.method()时,都调用Sub实例的method()方法,此时父类Base的实例方法method()被子类Sub覆盖。
引用变量sub1被声明为Base类型,Java虚拟机在执行sub1.staticMethod()时,调用Base类的staticMethod()方法,可见父类Base的静态方法staticMethod()不能被子类覆盖。
引用变量sub2被声明为Sub类型,Java虚拟机在执行sub2.staticMethod()时,调用Sub类的staticMethod()方法,Base类的staticMethod()方法被Sub类staticMethod()方法隐藏。
7、父类的非静态方法不能被子类覆盖为静态方法。
例如以下代码是不合法的:
public class Base{
void method(){}
}
public class Sub{
static void method(){} //编译出错,不能把父类实例方法覆盖为静态方法
}
8、父类的私有方法不能被子类覆盖。
下面的例子中,子类Sub中定义了一个和父类Base中方法同名、参数签名和返回类型一致,但访问权限不一致的方法showMe(),父类中showMe()的访问权限为private,而子类中showMe()的访问权限为public。尽管这在形式上和覆盖很相似,但Java虚拟机对此有不同的处理机制。子类方法覆盖父类方法的前提是,子类必须能继承父类的特定方法,由于Base类的private访问权限的showMe()方法不能被Sub类继承,因此Base类的showMe()方法和Sub类的showMe()方法之间并没有覆盖关系。
class Base{
private String showMe(){
return "Base";
}
public void print(){
System.out.println(showMe()); //到底调用Base实例的showMe()还是Sub类的showMe()?
}
}
public class Sub extends Base{
public String showMe(){
return "Sub";
}
public static void main(String[] args){
Sub sub = new Sub();
sub.print();
}
}
执行以上Sub类的main方法,会打印结果“Base”,这是因为print()方法在Base类中定义,因此print()方法会调用在Base类中定义的private访问权限的showMe()方法。
但是如果把Base类的showMe()方法改为public访问权限,其他代码不变。再执行以上Sub类的main()方法代码,会打印结果“Sub”,这是因为此时Sub类的showMe()方法覆盖了Base类的showMe()方法。因此尽管print()方法在Base类定义,Java虚拟机还是会调用当前的Sub实例的showMe()方法。
9、父类的抽象方法可以被子类通过两种途径覆盖:一是子类实现父类的抽象方法;二是子类重新声明父类的抽象方法。
例如以下代码合法:
public abstract class Base{
abstract void method1();
abstract void method2();
}
public abstract class Sub extends Base{
public void method1(){} //实现父类的抽象方法method1(),并且扩大了访问权限
public abstract void method2(); //重新声明method2()方法,仅仅扩大访问权限,但不实现
}
说明:侠义的理解,覆盖仅指子类覆盖父类的具体方法,即非抽象方法,在父类中提供了方法的默认实现方式,而子类采用不同的实现方式。我们为了叙述方便,也把子类实现父类的抽象方法也看作方法覆盖。
例如以下代码不合法:
public abstract class Base{
abstract void method1();
abstract void method2();
}
public abstract class Sub extends Base{
private void method1(){} //编译出错,不能缩小访问权限
private abstract void method2(); //编译出错,不能缩小访问权限
}
10、父类的非抽象方法可以覆盖为抽象方法。
例如以下代码合法:
public class Base{
void method(){}
}
public class Sub extends Base{
public abstract void method(); //合法
}
分享到:
相关推荐
《JAVA面向对象编程》是IT领域内一部权威的教材,由资深软件工程师孙卫琴编著,本书系统地介绍了JAVA语言的面向对象编程思想与技术。面向对象编程(Object-Oriented Programming,简称OOP)是一种编程范式,它将程序...
综上所述,《JAVA面向对象编程_孙卫琴2.pdf》这本书通过理论结合实践的方式,深入浅出地介绍了面向对象编程的基本原理和技术要点。无论是对于初学者还是有一定经验的开发者来说,都是一本不可多得的好书。通过对本书...
JAVA 面向对象编程 孙卫琴
孙卫琴在其著作《Java面向对象编程》中提到了多个重要的设计原则: - **单一职责原则(Single Responsibility Principle, SRP)**:一个类应该只有一个引起它变化的原因。这意味着每个类应该专注于实现单一的功能。...
根据提供的文件信息,关于《JAVA面向对象编程(孙卫琴)01》这本书,我们可以提炼出以下知识点: 首先,这本书是由孙卫琴编著的,专门针对JAVA面向对象编程的教程书籍。从标题我们可以知道,这本书是面向对象编程的...
1. Java面向对象编程:面向对象编程(Object-Oriented Programming,简称OOP)是一种编程范式,依赖于对象的概念来设计应用程序。在面向对象编程中,数据和方法被封装成一个单一的单元,即对象。这些对象通过发送...
根据提供的文件信息,“JAVA面向对象编程(孙卫琴)”是一本深入浅出介绍Java面向对象编程的书籍。这本书被广泛认为是学习Java语言的理想入门教材之一,尤其适合那些希望掌握面向对象编程基本概念和技术的新手。 ### ...
本书内容由浅入深,紧密结合实际,利用大量典型实例,详细讲解Java面向对象的编程思想、编程语法和设计模式,介绍常见Java类库的用法,总结优化Java编程的各种宝贵经验,深入阐述Java虚拟机执行Java程序的原理。...
Java面向对象编程是Java开发中的核心概念,由孙卫琴老师讲解的这门课程深入浅出地阐述了这一主题。源代码是学习编程时的重要参考资料,它提供了实践示例,帮助学习者理解并掌握理论知识。这个压缩包包含了孙卫琴老师...
《Java面向对象编程》是Java学习的经典教材,由知名讲师孙卫琴编著。这份资料包含PPT演示文稿和配套的源代码,旨在帮助学习者深入理解和掌握Java的面向对象编程思想。以下是对其中主要知识点的详细阐述: 1. **面向...
JAVA面向对象编程孙卫琴java流程控制.pdf
本书采用由浅入深、与实际应用紧密结合的方式,利用大量经典的实例,详细讲解Java面向对象的编程思想、编程语法和设计模式,介绍常见Java类库的用法,总结优化Java编程的各种宝贵经验,深入阐述Java虚拟机执行Java...
《孙卫琴《Java面向对象编程》配套PPT》是一份深入浅出的教程资源,旨在帮助学习者全面掌握Java编程中的面向对象概念和技术。Java作为一种广泛应用的编程语言,其面向对象特性是其核心魅力所在,它使得代码更加模块...
JAVA面向对象编程_孙卫琴 313M pdf 共14部分
JAVA面向对象编程(孙卫琴)(带完整书签版,课后答案,源代码) 最好的JAVA入门书籍 讲解全面,透彻 需要单独下载课后答案和源代码的在本人资源中可以找到,欢迎下载 本压缩包共分为6个部分,这里为第二部分。
《JAVA面向对象编程》是孙卫琴老师的一本经典教材,深入浅出地讲解了Java语言的核心特性,尤其是面向对象编程的理念与实践。这本教材的示例源代码提供了丰富的学习资源,帮助读者通过实际操作理解理论知识。 首先,...
JAVA面向对象编程(孙卫琴)(带完整书签版,课后答案,源代码) 最好的JAVA入门书籍 讲解全面,透彻 需要单独下载课后答案和源代码的在本人资源中可以找到,欢迎下载 本压缩包共分为6个部分,这里为第三部分。
Java面向对象编程是Java语言的核心特性之一,由孙卫琴老师讲授的这门课程深入浅出地介绍了这一主题。课后答案集包含了对课程中习题的详细解答,是学习者巩固理论知识、提升实践技能的重要参考资料。下面将详细阐述...