java override例子_Java 重写(Override)与重载(Overload)

重写(Override)

重写是子类对父类的允许访问的方法的实现过程进行重新编写, 返回值和形参都不能改变。即外壳不变,核心重写!

重写的好处在于子类可以根据需要,定义特定于自己的行为。 也就是说子类能够根据需要实现父类的方法。

重写方法不能抛出新的检查异常或者比被重写方法申明更加宽泛的异常。例如: 父类的一个方法申明了一个检查异常 IOException,但是在重写这个方法的时候不能抛出 Exception 异常,因为 Exception 是 IOException 的父类,只能抛出 IOException 的子类异常。

在面向对象原则里,重写意味着可以重写任何现有方法。实例如下:

TestDog.java 文件代码:

1 classAnimal{2 public voidmove(){3 System.out.println("动物可以移动");4 }5 }6

7 class Dog extendsAnimal{8 public voidmove(){9 System.out.println("狗可以跑和走");10 }11 }12

13 public classTestDog{14 public static voidmain(String args[]){15 Animal a = new Animal(); //Animal 对象

16 Animal b = new Dog(); //Dog 对象

17

18 a.move();//执行 Animal 类的方法

19

20 b.move();//执行 Dog 类的方法

21 }22 }

以上实例编译运行结果如下:

动物可以移动

狗可以跑和走

在上面的例子中可以看到,尽管b属于Animal类型,但是它运行的是Dog类的move方法。

这是由于在编译阶段,只是检查参数的引用类型。

然而在运行时,Java虚拟机(JVM)指定对象的类型并且运行该对象的方法。

因此在上面的例子中,之所以能编译成功,是因为Animal类中存在move方法,然而运行时,运行的是特定对象的方法。

思考以下例子:

TestDog.java 文件代码:

1 classAnimal{2 public voidmove(){3 System.out.println("动物可以移动");4 }5 }6

7 class Dog extendsAnimal{8 public voidmove(){9 System.out.println("狗可以跑和走");10 }11 public voidbark(){12 System.out.println("狗可以吠叫");13 }14 }15

16 public classTestDog{17 public static voidmain(String args[]){18 Animal a = new Animal(); //Animal 对象

19 Animal b = new Dog(); //Dog 对象

20

21 a.move();//执行 Animal 类的方法

22 b.move();//执行 Dog 类的方法

23 b.bark();24 }25 }

以上实例编译运行结果如下:

1 TestDog.java:30: cannot find symbol2 symbol : method bark()3 location: classAnimal4 b.bark();5 ^

该程序将抛出一个编译错误,因为b的引用类型Animal没有bark方法。

方法的重写规则

参数列表必须完全与被重写方法的相同。

返回类型与被重写方法的返回类型可以不相同,但是必须是父类返回值的派生类(java5 及更早版本返回类型要一样,java7 及更高版本可以不同)。

访问权限不能比父类中被重写的方法的访问权限更低。例如:如果父类的一个方法被声明为 public,那么在子类中重写该方法就不能声明为 protected。

父类的成员方法只能被它的子类重写

声明为 final 的方法不能被重写。

声明为 static 的方法不能被重写,但是能够被再次声明。

子类和父类在同一个包中,那么子类可以重写父类所有方法,除了声明为 private 和 final 的方法。

子类和父类不在同一个包中,那么子类只能够重写父类的声明为 public 和 protected 的非 final 方法。

重写的方法能够抛出任何非强制异常,无论被重写的方法是否抛出异常。但是,重写的方法不能抛出新的强制性异常,或者比被重写方法声明的更广泛的强制性异常,反之则可以。

构造方法不能被重写。

如果不能继承一个方法,则不能重写这个方法。

Super 关键字的使用

当需要在子类中调用父类的被重写方法时,要使用 super 关键字。

TestDog.java 文件代码:

1 classAnimal{2 public voidmove(){3 System.out.println("动物可以移动");4 }5 }6

7 class Dog extendsAnimal{8 public voidmove(){9 super.move(); //应用super类的方法

10 System.out.println("狗可以跑和走");11 }12 }13

14 public classTestDog{15 public static voidmain(String args[]){16

17 Animal b = new Dog(); //Dog 对象

18 b.move(); //执行 Dog类的方法

19

20 }21 }

以上实例编译运行结果如下:

动物可以移动

狗可以跑和走

重载(Overload)

重载(overloading) 是在一个类里面,方法名字相同,而参数不同。返回类型可以相同也可以不同。

每个重载的方法(或者构造函数)都必须有一个独一无二的参数类型列表。

最常用的地方就是构造器的重载。

重载规则:

被重载的方法必须改变参数列表(参数个数或类型不一样);

被重载的方法可以改变返回类型;

被重载的方法可以改变访问修饰符;

被重载的方法可以声明新的或更广的检查异常;

方法能够在同一个类中或者在一个子类中被重载。

无法以返回值类型作为重载函数的区分标准。

实例

Overloading.java 文件代码:

1 public classOverloading {2 public inttest(){3 System.out.println("test1");4 return 1;5 }6

7 public void test(inta){8 System.out.println("test2");9 }10

11 //以下两个参数类型顺序不同

12 public String test(inta,String s){13 System.out.println("test3");14 return "returntest3";15 }16

17 public String test(String s,inta){18 System.out.println("test4");19 return "returntest4";20 }21

22 public static voidmain(String[] args){23 Overloading o = newOverloading();24 System.out.println(o.test());25 o.test(1);26 System.out.println(o.test(1,"test3"));27 System.out.println(o.test("test4",1));28 }29 }

重写与重载之间的区别

区别点重载方法重写方法

参数列表

必须修改

一定不能修改

返回类型

可以修改

一定不能修改

异常

可以修改

可以减少或删除,一定不能抛出新的或者更广的异常

访问

可以修改

一定不能做更严格的限制(可以降低限制)

总结

方法的重写(Overriding)和重载(Overloading)是java多态性的不同表现,重写是父类与子类之间多态性的一种表现,重载可以理解成多态的具体表现形式。

(1)方法重载是一个类中定义了多个方法名相同,而他们的参数的数量不同或数量相同而类型和次序不同,则称为方法的重载(Overloading)。

(2)方法重写是在子类存在方法与父类的方法的名字相同,而且参数的个数与类型一样,返回值也一样的方法,就称为重写(Overriding)。

(3)方法重载是一个类的多态性表现,而方法重写是子类与父类的一种多态性表现。

87f17032a89d5bee91bb3d5d5ec773d8.png

为什么函数不能根据返回类型来区分重载?(来自华为面试题)

答:

因为在调用时不能指定类型信息,编译器不知道你要调用哪个函数

例如:

1 float max(int a,intb);2 int max(int a,int b);

当调用max(1,2)时无法确定抵用的是哪个,单从这一点来说,仅返回值类型不同的重载是不应允许的。

在比如对下面这两个方法来说,虽然它们有同样的名字和自变量,但其实是很容易区分的

1 voidf () {}2 int f () {}

若编译器可以根据上下文明确判断出含义,比如在int x = f()中,那么这样做完全没有问题。然而,我们也可能调用一个方法,同时忽略返回值;我们通常把这称为“为它的副作用去调用一个方法”,因为我们关心的不是返回值,而是方法调用的其他效果。所以假如我们像下面这样调用方法: f(); Java 怎样判断 f()的具体调用方式呢?而且别人如何识别并理解代码呢?由于存在这一类的问题,所以不能。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值