Java——重写(Override)总结
重写(Override)
重写是子类对父类的允许访问的方法的实现过程进行重新编写!返回值和形参都不能改变。即外壳不变,核心重写!
重写的好处在于子类可以根据需要,定义特定于自己的行为。也就是说子类能够根据需要实现父类的方法。
方法重写的规则
- 参数列表必须完全与被重写方法的相同;
- 返回类型必须完全与被重写方法的返回类型相同;
- 访问权限不能比父类中被重写的方法的访问权限更高。例如:如果父类的一个方法被声明为public,那么在子类中重写该方法就不能声明为protected。
- 父类的成员方法只能被它的子类重写。
- 声明为final的方法不能被重写。
- 声明为static的方法不能被重写,但是能够被再次声明。
- 如果一个方法不能被继承,那么该方法不能被重写。
- 子类和父类在同一个包中,那么子类可以重写父类所有方法,除了声明为private和final的方法。
- 子类和父类不在同一个包中,那么子类只能够重写父类的声明为public和protected的非final方法。
- 重写的方法能够抛出任何非强制异常,无论被重写的方法是否抛出异常。但是,重写的方法不能抛出新的强制性异常,或者比被重写方法声明的更广泛的强制性异常,反之则可以。
- 构造方法不能被重写。
- 如果不能继承一个方法,则不能重写这个方法。
重载(Overload)
重载(overloading) 是在一个类里面,方法名字相同,而参数不同。返回类型可以相同也可以不同。
每个重载的方法(或者构造函数)都必须有一个独一无二的参数类型列表。
只能重载构造函数
方法重载的规则
- 被重载的方法必须改变参数列表;
- 被重载的方法可以改变返回类型;
- 被重载的方法可以改变访问修饰符;
- 被重载的方法可以声明新的或更广的检查异常;
- 方法能够在同一个类中或者在一个子类中被重载
方法重写与重载的区别
区别点 | 重载方法 | 重写方法 |
---|---|---|
参数列表 | 必须修改 | 一定不能修改 |
返回类型 | 可以修改 | 一定不能修改 |
异常 | 可以修改 | 可以减少或删除,一定不能抛出新的或者更广的异常 |
访问 | 可以修改 | 一定不能做更严格的限制(可以降低限制) |
相关练习题
1 重载的方法能否根据返回类型进行区分?
1)方法的重载和重写都是实现多态的方式,区别在于前者实现的是编译时的多态性,而后者实现的是运行时的多态性。
2)重载发生在一个类中,同名的方法如果有不同的参数列表(参数类型不同、参数个数不同或者二者都不同)则视为重载。
3)重写发生在子类与父类之间,重写要求子类被重写方法与父类被重写方法有相同的返回类型,比父类被重写方法更好访问,不能比父类被重写方法声明更多的异常(里氏代换原则)。
4)重载对返回类型没有特殊的要求。
2 写出一段示例代码展示方法重写和重载的区别
重写:
public class OverrideDemo {
public static void main(String[] args) {
Animal animal = new Animal();
animal.animalPrint();
Animal cat = new Cat();
cat.animalPrint();
Animal dog = new Dog();
dog.animalPrint();
}
}
class Animal {
public void animalPrint() {
System.out.println("这是动物");
}
}
class Cat extends Animal {
@Override
public void animalPrint() {
System.out.println("这是猫");
}
}
class Dog extends Animal {
@Override
public void animalPrint() {
System.out.println("这是狗狗");
}
}
运行结果:
这是动物
这是猫
这是狗狗
Process finished with exit code 0
重载
public class OverrideDemoTest {
public void printDemo() {
System.out.println("无参方法");
}
public void printDemo(int a) {
System.out.println("一个参数的方法,参数为:" + a);
}
public void printDemo(int a, String b) {
System.out.println("两个参数的方法,第一个参数为:" + a + " 第二个参数为:" + b);
}
public static void main(String[] args) {
OverrideDemoTest overrideDemoTest = new OverrideDemoTest();
overrideDemoTest.printDemo();
overrideDemoTest.printDemo(1);
overrideDemoTest.printDemo(1, "Hello");
}
}
运行结果:
无参方法
一个参数的方法,参数为:1
两个参数的方法,第一个参数为:1 第二个参数为:Hello
Process finished with exit code 0
习题:
1.abstract class Name {
private String name;
public abstract boolean isStupidName(String name) {}
}
错。abstract method必须以分号结尾,且不带花括号。
2.public class Something {
void doSomething() {
private String s = “”;
int l = s.length();
}
}
错。局部变量前不能放置任何访问修饰符 (private,public,和protected)。final可以用来修饰局部变量
(final如同abstract和strictfp,都是非访问修饰符,strictfp只能修饰class和method而非variable)。
3.abstract class Something {
private abstract String doSomething ();
}
错。abstract的methods不能以private修饰。abstract的methods就是让子类implement(实现)具体细节的,怎么可以用private把abstract method*起来呢? (同理,abstract method前不能加final)。
4.public class Something {
public int addOne(final int x) {
return ++x;
}
}
错。int x被修饰成final,意味着x不能在addOne method中被修改。
5.public class Something {
public static void main(String[] args) {
Other o = new Other();
new Something().addOne(o);
}
public void addOne(final Other o) {
o.i++;
}
}
class Other {
public int i;
}
正确。在addOne method中,参数o被修饰成final。如果在addOne method里我们修改了o的reference
(比如: o = new Other()????,那么如同上例这题也是错的。但这里修改的是o的member vairable
(成员变量),而o的reference并没有改变。
1.(多选题)给定java代码如下所示,在1处新增下列()方法,是对show()方法的重载
public class Test{
public void show(int x, int y, int z) { }
//1
}
A.public int show(int x,int y,float z){return 0;}
B.public int show(int x,int y,int z){return 0;}
C.public void show(int x,int z){}
D.public viod show(int z,int y,int x){}
答案:AC
解析:本题考查方法的重载规则
方法的重载必须满足方法名相同和参数不同(类型或个数)
选项AC满足重载的条件,正确
选项BD参数都与定义的方法参数相同,不是重载。
2.(单选题)类A、B和C的定义如下:
public class A {
public void f() {
System.out.println("A.f()");
}
}
public class B extends A {
public void f() {
System.out.println("B.f()");
}
}
public class C {
public void g(A a) {
System.out.println("g(A a)");
a.f();
}
public void g(B b) {
System.out.println("g(B b)");
b.f();
}
}
运行下列程序:
C c = new C();
A e = new B();
c.g(e);
输出的结果是()
A. g(A a) A.f()
B. g(A a) B.f()
C. g(B b) A.f()
D. g(B b) B.f()
答案:B
解析:C类中的两个方法是重载的方法,编译器会在编译时根据签名的不同来绑定调用不同的方法
A e =new B(); 这句是一个向上造型,e是A类的引用,但是指向的是B类的对象,而重载看的是参数的引用类型,重写才是看的对象类型
因为c调用的g方法参数是A类型的,所以匹配的是g(A a)的方法,先输出"g(A a)",然后执行a.f();
A类和B类是父子类关系,两个f()是重写的方法,执行哪个方法类的方法要看具体的对象
而e指向的对象是B对象,所以找到B类中的f方法输出"B.f()"
也就是子类继承父类时,调用的方法都是子类重写后的方法
3.(多选题)请看下列代码
public class Tetromino {
protected int move(int x) {
return 0;
}
}
class T extends Tetromino {
<插入代码>
}
在<插入代码>处填入选项中的代码,使T类没有编译错误的是()
A.public int move(int x) { return 0; }
B.private int move(int x) { return 0; }
C.private int move(long x) { return 0; }
D.protected long move(int x) { return 0; }
答案:AC
解析: 本题考查方法的重写和重载语法。
注:方法的重写要遵循“两同两小一大”规则,“两同”即方法名相同,形参列表相同;“两小”指的是子类方法返回值类型应比父类方法返回值类型更小或相等;子类方法声明抛出的异常类应比父类方法声明抛出的异常类更小或相等;“一大”指的是子类方法的访问权限应比父类方法的访问范围更大或相等。
选项A,符合重写的定义,A选项正确。
选项B,不符合重写也不符合重载的语法(访问权限为private小于父类的protected,不满足重写;参数列表相同,不满足重载),B选项错误。
选项C,符合重载的语法,子类T继承父类Tetromino的 protected int move(int x) 方法,而C选择属于对重写方法的重载,所以C选择正确。
选项D,不符合重写也不符合重载的语法(返回值类型为long,与父类的int不同,不满足重写;参数列表相同,不满足重载),D选项错误
每日笔记:
重写和重载有什么区别?重写的两同两小一大规则是什么?
Overload是重载的意思,Override是覆盖的意思,也就是重写。
重载Overload表示同一个类中可以有多个名称相同的方法,但这些方法的参数列表各不相同(即参数个数或类型不同)。
重写Override表示子类中的方法可以与父类中的某个方法的名称和参数完全相同,通过子类创建的实例对象调用这个方法时,将调用子类中的定义方法,这相当于把父类中定义的那个完全相同的方法给覆盖了,这也是面向对象编程的多态性的一种表现。
重写要遵循"两同两小一大"原则:
两同:1)方法名相同 2)参数列表相同
两小: 1)子类方法的返回值类型小于或等于父类的
1.1)void时必须相等
1.2)基本类型时必须相等
1.3)引用类型时小于或相等:父类大,子类小
2)子类抛出的异常小于或等于父类的
一大:子类方法的访问权限大于或等于父类的