java方法覆盖的原则

什么是方法覆盖

如果在子类中定义的一个方法,其名称、返回类型及参数签名正好与父类中某个方法的名称、返回类型及参数签名相匹配,那么可以说,子类的方法覆盖了父类的方法。

覆盖方法必须满足的十大约束
[color=red]
一:子类方法的名称、参数签名和返回类型必须与父类方法的名称、参数签名和返回类型一致[/color]
[color=red]
二:子类方法不能缩小父类方法的访问权限[/color]
[color=red]
三:子类方法不能抛出比父类方法更多的异常,子类方法抛出的异常必须和父类方法抛出的异常相同,或者子类方法抛出的异常类是父类方法抛出的异常类的子类[/color]

[color=red]四:方法覆盖只存在于子类和父类(包括直接父类和间接父类)之间。在同一个类中方法只能被重载,不能被覆盖。[/color]

[color=red]五:父类的静态方法不能被子类覆盖为非静态方法。
(子类的非静态方法不能覆盖父类的静态方法)[/color]
[color=red]
六:子类可以定义与父类的静态方法同名的静态方法,以便在子类中隐藏父类的静态方法。[/color]

子类隐藏父类的静态方法和子类覆盖父类的实例方法,这两者的区别在于:运行时,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

[color=red]七:父类的非静态方法不能被子类覆盖为静态方法。
(父类的非静态方法不能被子类的静态方法所覆盖)[/color]

[color=red]八:父类的私有方法不能被子类覆盖。[/color]
同第六点一样,再来个经典的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();
}
}


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

[color=red]十:父类的非抽象方法可以被覆盖为抽象方法。[/color]



另外,关于构造函数:

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

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(){
//
}
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值