重写(override)与重载(overload)、虚方法、强制性异常
备注:若有不正之处,请多谅解并欢迎批评指正。转载请标明链接:https://www.cnblogs.com/pmbb/p/11409443.html
重写(Override)
重写是子类对父类的允许访问的方法的实现过程进行重新编写, 返回值和形参都不能改变。即外壳不变,核心重写!
重写的好处在于子类可以根据需要,定义特定于自己的行为。 也就是说子类能够根据需要实现父类的方法。
重写方法不能抛出新的检查异常或者比被重写方法申明更加宽泛的异常。例如: 父类的一个方法申明了一个检查异常 IOException,但是在重写这个方法的时候不能抛出 Exception 异常,因为 Exception 是 IOException 的父类,只能抛出 IOException 的子类异常。
在面向对象原则里,重写意味着可以重写任何现有方法。实例如下:
TestDog.java 文件代码:
1 class Animal{ 2 public void move(){ 3 System.out.println("动物可以移动"); 4 } 5 } 6 7 class Dog extends Animal{ 8 public void move(){ 9 System.out.println("狗可以跑和走"); 10 } 11 } 12 13 public class TestDog{ 14 public static void main(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 class Animal{ 2 public void move(){ 3 System.out.println("动物可以移动"); 4 } 5 } 6 7 class Dog extends Animal{ 8 public void move(){ 9 System.out.println("狗可以跑和走"); 10 } 11 public void bark(){ 12 System.out.println("狗可以吠叫"); 13 } 14 } 15 16 public class TestDog{ 17 public static void main(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 }
以上实例编译运行结果如下:
TestDog.java:30: cannot find symbol
symbol : method bark()
location: class Animal
b.bark();
^
该程序将抛出一个编译错误,因为b的引用类型Animal没有bark方法。
如果你想运行bark()方法的话,把
b.bark();
改成
Dog c=(Dog)b;//向下转行
c.bark();
即可
方法的重写规则
参数列表必须完全与被重写方法的相同。
返回类型与被重写方法的返回类型可以不相同,但是必须是父类返回值的派生类(java5 及更早版本返回类型要一样,java7 及更高版本可以不同)。
访问权限不能比父类中被重写的方法的访问权限更低。例如:如果父类的一个方法被声明为 public,那么在子类中重写该方法就不能声明为 protected。
父类的成员方法只能被它的子类重写。
声明为 final 的方法不能被重写。
声明为 static 的方法不能被重写,但是能够被再次声明。
子类和父类在同一个包中,那么子类可以重写父类所有方法,除了声明为 private 和 final 的方法。
子类和父类不在同一个包中,那么子类只能够重写父类的声明为 public 和 protected 的非 final 方法。
重写的方法能够抛出任何非强制异常,无论被重写的方法是否抛出异常。但是,重写的方法不能抛出新的强制性异常,或者比被重写方法声明的更广泛的强制性异常,反之则可以。
构造方法不能被重写。
如果不能继承一个方法,则不能重写这个方法。
Super 关键字的使用
当需要在子类中调用父类的被重写方法时,要使用 super 关键字。
1 class Animal{ 2 public void move(){ 3 System.out.println("动物可以移动"); 4 } 5 } 6 7 class Dog extends Animal{ 8 public void move(){ 9 super.move(); // 应用super类的方法 10 System.out.println("狗可以跑和走"); 11 } 12 } 13 14 public class TestDog{ 15 public static void main(String args[]){ 16 17 Animal b = new Dog(); // Dog 对象 18 b.move(); //执行 Dog类的方法 19 20 } 21 }
以上实例编译运行结果如下:
动物可以移动 狗可以跑和走
重载(Overload)
重载(overloading) 是在一个类里面,方法名字相同,而参数不同。返回类型可以相同也可以不同。
每个重载的方法(或者构造函数)都必须有一个独一无二的参数类型列表。
最常用的地方就是构造器的重载。
重载规则:
- 被重载的方法必须改变参数列表(参数个数或类型不一样);
- 被重载的方法可以改变返回类型;
- 被重载的方法可以改变访问修饰符;
- 被重载的方法可以声明新的或更广的检查异常;
- 方法能够在同一个类中或者在一个子类中被重载。
- 无法以返回值类型作为重载函数的区分标准。
实例
Overloading.java 文件代码:
1 public class Overloading { 2 public int test(){ 3 System.out.println("test1"); 4 return 1; 5 } 6 7 public void test(int a){ 8 System.out.println("test2"); 9 } 10 11 //以下两个参数类型顺序不同 12 public String test(int a,String s){ 13 System.out.println("test3"); 14 return "returntest3"; 15 } 16 17 public String test(String s,int a){ 18 System.out.println("test4"); 19 return "returntest4"; 20 } 21 22 public static void main(String[] args){ 23 Overloading o = new Overloading(); 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)方法重载是一个类的多态性表现,而方法重写是子类与父类的一种多态性表现。
虚方法
我们先了解什么是“基类”?在面向对象设计中,被定义为包含所有实体共性的class类型,被称为“基类”。
什么是派生类?利用继承机制,新的类可以从已有的类中派生。那些用于派生的类称为这些特别派生出的类的“基类”。
什么是虚方法?
虚方法可以有实现体,若一个实例方法的声明中含有 virtual 修饰符,则称该方法为虚方法。使用了 virtual 修饰符后,不允许再有 static、abstract 或者 override 修饰符。
简单工厂模式中虚方法代码:
1 //用于运算后的结果 2 public virtual double GetResult() 3 { 4 double result = 0; 5 return result; 6 }
可以看出该类中有一个 GetResult() 的方法,GetResult() 方法中带有一个 virtual 修饰符,该修饰符表明:该基类的派生类可以重载该方法。 GetResult() 方法的作用:输出语句 "这是一个虚方法!" 到控制台。
一个虚方法的实现可以由派生类取代。取代所继承的虚方法的实现的过程称为重写该方法;在一个虚方法调用中,该调用所设计的那个实例运行时的类型确定了要被调用的究竟是该方法的哪一个实现。
抽象方法和虚方法的区别
- 1.虚方法必须有实现部分,抽象方法没有提供实现部分,抽象方法是一种强制派生类覆盖的方法,否则派生类将不能被实例化。
- 2.抽象方法只能在抽象类中声明,虚方法不是。如果类包含抽象方法,那么该类也是抽象的,也必须声明类是抽象的。
- 3.抽象方法必须在派生类中重写,这一点和接口类似,虚方法不需要再派生类中重写。
简单说,抽象方法是需要子类去实现的。虚方法是已经实现了的,可以被子类覆盖,也可以不覆盖,取决于需求。
抽象方法和虚方法都可以供派生类重写。
强制性异常
java中的异常分为两大类,强制性异常(CheckedException)和非强制性异常(UncheckedException)。而java中除了运行时异常(RuntimeException)外,都是强制性异常。
非强制性异常:所谓非强制性异常就和上面相反了。不过你当然也可以try catch或者thows,只不过这不是强制性的。
参考网址:
重写与重载:https://www.runoob.com/java/java-override-overload.html
虚方法:https://www.runoob.com/w3cnote/what-is-a-virtual-method.html
强制性异常:https://blog.csdn.net/qq_39536158/article/details/83093781