方法的重写(override/overwrite)
@[toc]
基础理解
- 定义:在子类中可以根据需要对从父类中继承来的方法进行改造,也称 为方法的重置、覆盖。在程序执行时,子类的方法将覆盖父类的方法。
代码举例:
- 首先写了Person类,然后写Student类继承,最后有一个Test用于测试
- Person类
package ChongXie;
public class Person {
String name;
int age;
public Person(){
}
public Person(String name,int age){
this.name=name;
this.age=age;
}
public void eat(){
System.out.println("吃饭");
}
public void walk(int s){
System.out.println("走路,走了"+s+"公里");
}
}
- Student类(未重写)
package ChongXie;
public class Student extends Person {
String major;
public Student(){
}
public Student(String major){
this.major=major;
}
//父类的继承,如何写子类自己的方法
public void study(){
System.out.println("学习,专业是:"+major);
}
//对父类声明的方法重写:在子类中可以根据需要对从父类中继承来的方法进行改造,
// 也称为方法的重置、覆盖。在程序执行时,子类的方法将覆盖父类的方法。
}
- Test测试类
package ChongXie;
public class Test {
public static void main(String[] args) {
Student s1 = new Student("计算机科学与技术");
s1.eat();
s1.walk(10);
s1.study();//上面是调用Student的方法
//方法的重写:在子类中可以根据需要对从父类中继承来的方法进行改造,
// 也称为方法的重置、覆盖。在程序执行时,子类的方法将覆盖父类的方法。
}
}
运行结果:
- 重写:(重写子类Student里面的eat方法)
package ChongXie;
public class Student extends Person {
String major;
public Student(){
}
public Student(String major){
this.major=major;
}
//父类的继承,如何写子类自己的方法
public void study(){
System.out.println("学习,专业是:"+major);
}
//对父类声明的方法重写:在子类中可以根据需要对从父类中继承来的方法进行改造,
// 也称为方法的重置、覆盖。在程序执行时,子类的方法将覆盖父类的方法。
//对父类中的eat方法进行了重写
public void eat(){
System.out.println("学生应该多吃有营养的食物");
}
}
重写之后Test运行结果:(除了对Student类的重写,其他的代码不变)
- 重写的应用:重写以后,当前创建子类对象以后,通过子类对象调用父类中同名同参的方法时,实际执行的是子类重写父类的方法
(当然,去创建父类对象,调用的重写过的方法,还是以前的方法不变)
重写的规定(重点)
方法的声明: 权限修饰符 返回值类型 方法名(){
//方法体
}
约定俗成:子类中的叫重写的方法,父类中的叫被重写的方法
- 要求:
- 子类重写的方法必须和父类被重写的方法具有相同的方法名称、参数列表
- 子类重写的方法的返回值类型不能大于父类被重写的方法的返回值类型
- 父类被重写的方法返回值类型是void,则子类重写方法的返回值也只能是void
- 父类被重写的方法的返回值类型是A类型,则子类重写的方法的返回值类型可以是A类或者A类的子类(所有类都是object类的子类)
- 如果返回值类型是基本数据类型的情况下:则子类重写的返回值类型,必须是相同的基本数据类型
- 子类重写的方法使用的访问权限不能小于父类被重写的方法的访问权限(重写的方法的权限修饰符不小于被重写的方法)
- 特殊情况:子类不能重写父类中声明为private权限的方法
- 子类方法抛出的异常不能大于父类被重写方法的异常(异常后续单独写出来)
ps:关于子类不能重写父类中声明为private权限的方法的验证和解释
举例子:父类person类,子类student类
父类写了一个方法
private void show(){
System.out.println("我是一个人");
}
子类写了一个方法
public void show(){
System.out.println("我是一个学生");
}
这里的子类写的方法不是对父类的重写,而是新写的一个方法,子类不能重写父类私有的方法
验证:
这里将父类的show方法的权限改为private
package ChongXie;
public class Person {
String name;
int age;
public Person(){
}
public Person(String name,int age){
this.name=name;
this.age=age;
}
public void eat(){
System.out.println("我是父类eat方法");
}
public void walk(int s){
System.out.println("走路,走了"+s+"公里");
show();
eat();
}
private void show(){
System.out.println("我是父类show方法");
}
}
然后子类对show方法进行”重写”(注意,这里并不是重写)
package ChongXie;
public class Student extends Person {
String major;
public Student(){
}
public Student(String major){
this.major=major;
}
//父类的继承,如何写子类自己的方法
public void study(){
System.out.println("我是子类study方法,专业是:"+major);
}
//对父类声明的方法重写:在子类中可以根据需要对从父类中继承来的方法进行改造,
// 也称为方法的重置、覆盖。在程序执行时,子类的方法将覆盖父类的方法。
//对父类中的eat方法进行了重写
public void eat(){
System.out.println("我是子类eat方法");
}
//父类写了show方法但是是private,所以子类写出来不算重写
public void show(){
System.out.println("我是子类show方法");
}
}
然后test测试类对其检测
package ChongXie;
public class Test {
public static void main(String[] args) {
Student s1 = new Student("计算机科学与技术");//建子类的对象
s1.walk(10);
}
}
运行结果发现
解释:按道理说运行结果应该是子类的show方法,因为已经对既然子类对show已经“重写”,那么walk方法中调用show就应该是子类的show,但是仍然为父类,可以看出并没有重写,而且walk中调用eat方法的确是子类的,这才是正确重写
暴力重写:重写既然这么多规规矩矩和约束,以及各种各样都可能犯错的点,那么实际开发中通常使用暴力重写,也就是直接把父类的方法的命名,复制粘贴到子类,这样百分百不会发生错误。
注意:
子类与父类中同名同参数的方法必须同时声明为非static的(即为重写),或者同时声明为 static的不是重写。因为static方法是属于类的,子类无法覆盖父类的方法