首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 开发语言 > 编程 >

JAVA中的步骤覆盖/覆写的十大原则

2012-10-26 
JAVA中的方法覆盖/覆写的十大原则什么是方法覆盖如果在子类中定义的一个方法,其名称、返回类型及参数签名正

JAVA中的方法覆盖/覆写的十大原则

什么是方法覆盖
如果在子类中定义的一个方法,其名称、返回类型及参数签名正好与父类中某

个方法的名称、返回类型及参数签名相匹配,那么可以说,子类的方法覆盖了父

类的方法。

覆盖方法必须满足的十大约束

一:子类方法的名称、参数签名和返回类型必须与父类方法的名称、参数签名和

返回类型一致

二:子类方法不能缩小父类方法的访问权限

三:子类方法不能抛出比父类方法更多的异常,子类方法抛出的异常必须和父类

方法抛出的异常相同,或者子类方法抛出的异常类是父类方法抛出的异常类的子



四:方法覆盖只存在于子类和父类(包括直接父类和间接父类)之间。在同一个

类中方法只能被重载,不能被覆盖。(哈哈,这个是方法覆盖的定义吧)。

五:父类的静态方法不能被子类覆盖为非静态方法。

六:子类可以定义与父类的静态方法同名的静态方法,以便在子类中隐藏父类的

静态方法。子类隐藏父类的静态方法和子类覆盖父类的实例方法,这两者的区别

在于:运行时,Java虚拟机把静态方法和所属的类绑定,而把实例方法和所属的
实例绑定。
?? 第六点比较长,你是否真的理解了呢???
看个SCJP题库的经典例子吧,并说说为什么是这样的结果

复制内容到剪贴板代码:class Base {
???? void method() {
???????? System.out.println("method of Base");
???? }
???? static void staticMethod() {
???????? System.out.println("static method of Base");
???? }
}

class Sub extends Base {
???? void method() {
???????? System.out.println("method of Sub");
???? }
???? static void staticMethod() {
???????? System.out.println("static method of Sub");
???? }
}

public class Test2 {
???? public static void main(String[] args) {
???????? Base sub1 = new Sub();
???????? sub1.method();
???????? sub1.staticMethod();
???????? Sub sub2 = new Sub();
???????? sub2.method();
???????? sub2.staticMethod();
???? }
}
答案:method of Sub
static method of Base
method of Sub
static method of Sub

七:父类的非静态方法不能被子类覆盖为静态方法。

八:父类的私有方法不能被子类覆盖。 同第六点一样,再来个经典的SCJP题目

class Base {
private String showMe() {
return "Base";
}

public void print(){
System.out.println(showMe());
showMe();
}
}

public class Sub extends Base {
public String showMe(){
return "Sub";
}
}
?? public static void main(String args[]){

?? Sub sub=new Sub();

?? sub.print();
?? }
}

各位是否答对了呢??? 这就是传说中的SCJP题目,哈哈都是考语法的。 想起偶当年初中时学英语的情景了,老师整天唧唧呱呱分析那些语法个不停。我想大家请自己的初中英语老师来讲core java。SCJP保证可以考满分。


九:父类的抽象方法可以被子类通过两种途径覆盖:一是子类实现父类的抽象方法;二是子类重新声明父类的抽象方法。

十:父类的非抽象方法可以被覆盖为抽象方法。

?

另外,关于构造函数:

子类的实例化过程:子类中所有的构造函数会去访问父类中的空参数构造函数,那是因为每一个子类构造函数中的第一行都有一句隐式super语句。

构造函数不能继承。子类的构造函数可以通过super关键字显式调用父类中的构造函数。如果子类中的构造函数没有显式调用父类中的构造函数,编译器就会自动在子类的构造函数中调用父类中参数为空的构造函数。于是,当父类中没有参数为空的构造函数,而子类中又没有显示调用父类的其他构造函数,编译时就会报错。这一点需要特别注意。当父类中没有定义任何构造函数时,编译器就会为它指定一个参数为空的默认的构造函数;如果父类中定义了构造函数,那么编译器就不会为它指定一个参数为空的默认构造函数了。因此,如果某个类有可能成为其他类的父类,为了避免发生不必要的编译错误,最好为它编写一个参数为空的构造函数。

eg1.父类Sup中没有定义构造函数,编译程序将为它指定一个参数为空的默认构造函数。子类Sub中也没有定义构造函数,编译程序也会为它指定一个参数为空的默认的构造函数,并且会在这个默认的构造函数中调用父类的参数为空的构造函数。
public class Sub extends Sup{
??? //子类中没有定义构造函数
??? public static void main(String args[]){
??????? Sub sub=new Sub();
??? }
}

class Sup{
??? //父类中没有定义构造函数
}


eg2.父类Sup中没有定义构造函数,编译程序将为它指定一个参数为空的默认构造函数。子类定义了一个带整型参数的构造函数,在这个构造函数中子类没有显式调用父类的构造函数,所以编译器为在它里面调用父类中参数为空的构造函数。

public class Sub extends Sup{
??? //子类中定义类一个带整型变量参数的构造函数
??? public Sub(int i){
??????? //
??? }
??? public static void main(String args[]){
??????? Sub sub=new Sub(1);
??? }
}
class Sup{
??? //父类中没有定义构造函数
}
eg3.父类中定义了一个带整型参数的构造函数,因此编译器不再为它指定参数为空的默认的构造函数。子类中也定义了一个带整型参数的构造函数。编译时,编译器将试图在子类的构造函数中调用父类的参数为空的构造函数,但是父类中没有定义参数为空的构造函数,所以编译程序将会报错。排错的方法时在子类的构造函数中显示调用父类的构造函数,或者在父类中添加一个带空参数的构造函数。

public class Sub extends Sup{
??? //子类中定义类一个带整型变量参数的构造函数
??? public Sub(int i){
??????? //
??? }
??? public static void main(String args[]){
??????? Sub sub=new Sub(1);
??? }
}

class Sup{
??? //父类中定义了一个带整型参数的构造函数
??? public Sup(int i){
??????? //
??? }
}
?? 排错方法1:

public class Sub extends Sup{
??? //子类中定义类一个带整型变量参数的构造函数
??? public Sub(int i){
??????? super(i);//调用父类中的构造函数
??? }
??? public static void main(String args[]){
??????? Sub sub=new Sub(1);
??? }
}

class Sup{
??? //父类中定义了一个带整型参数的构造函数
??? public Sup(int i){
??????? //
??? }
}
?? 排错方法2:

public class Sub extends Sup{
??? //子类中定义类一个带整型变量参数的构造函数
??? public Sub(int i){
??????? //
??? }
??? public static void main(String args[]){
??????? Sub sub=new Sub(1);
??? }
}

class Sup{
??? //父类中定义了一个带整型参数的构造函数
??? public Sup(int i){
??????? //
??? }
??? //定义一个带空参数的构造函数
??? public Sup(){
??????? //
??? }
}

热点排行