子类从父类中继承方法。有时,子类需要修改父类中定义的方法的实现,这称作方法重写(method overriding)。
在这里先说三个小知识,一、Object是所有类的父类。 二、当我们每使用new 新建一个实例对象如Object o=new Object(); o所代表的是实例的引用地址,每使用new新建一个示例赋给o,o的地址就会改变一次。三、Object.equal(obj)方法判断的是对象的引用地址是否相同,相同返回true,不同返回false。如果不理解的话,就先记住,以后学java虚拟机的时候就理解了。如果是新手只需牢记第一点就行,二三了解即可。
public static void main(String[] args)
{
Object o1 = new Object();
Object o2 = new Object();
System.out.println(o1.equals(o2));
String s1 = new String();
String s2 = new String();
System.out.println(s1.equals(s2));
}
由上面的小知识可知,Object是String的父类。那么再结合上一章的继承知识,Object.equals(obj)和String.equals(obj)是相同的。String继承了Object的equlas方法。
现在我们来猜测一下两个输出语句的结果是否相同。如果你猜测的为两个false,那说明你理解了继承的概念,从目前的思路上来说这个回答没有问题。我们来看一下结果:
相信你一定猜到了两个输出语句的结果肯定是不同的,不然我也不会在前面说这么多铺垫了。在学习的过程中,眼前的结果不是最重要的,更重要的是知其所以然。话不多说,直接上源码:
Object.equals()的源码:
//别怀疑,这就是源码,只有一句。
public boolean equals(Object obj)
{
return (this == obj);
}
String.equals()的源码
public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String anotherString = (String) anObject;
int n = value.length;
if (n == anotherString.value.length) {
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}
对比代码发现父类的equal方法和子类的equal方法不一样。子类对于父类相同方法(参数、方法名、返回值均相同)重新编写叫做重写,string类的equal方法就是将object的equal方法进行重写,先对引用变量的地址值进行判断,如果判断地址不同,再对长度和内容进行对比.因为地址不同,所以o1.equals(o2)为false。再来看s1.equals(s2),现对s1和s2第引用地址进行判断,得到地址不同,则继续执行代码,再对其内容进行判断,判断结果相同则返回true。
方法重写的详细规则:
- 参数列表必须完全与被重写方法的相同;
- 返回类型必须完全与被重写方法的返回类型相同;
- 访问权限不能比父类中被重写的方法的访问权限更低。例如:如果父类的一个方法被声明为public,那么在子类中重写该方法就不能声明为protected。
- 父类的成员方法只能被它的子类重写。
- 声明为final的方法不能被重写。
- 声明为static的方法不能被重写,但是能够被再次声明。
- 子类和父类在同一个包中,那么子类可以重写父类所有方法,除了声明为private和final的方法。
- 子类和父类不在同一个包中,那么子类只能够重写父类的声明为public和protected的非final方法。
- 重写的方法能够抛出任何非强制异常,无论被重写的方法是否抛出异常。但是,重写的方法不能抛出新的强制性异常,或者比被重写方法声明的更广泛的强制性异常,反之则可以。
- 构造方法不能被重写。
- 如果不能继承一个方法,则不能重写这个方法。