尽量不要为了复用代码而使用继承。具体指,别人在其他包里开发的类,此时为了复用直接继承。也就是那个类的开发维护并不是由我们来掌控。
这是因为由于自用性的存在,我们必须十分了解父类的实现,而且,父类随时可能更改,子类的一些假设会被推翻。
什么是自用性?
就是父类里面的方法调用了其他父类里面的方法,这个现象在继承的时候会有问题。如果父类的一个方法f使用了一个可以被子类覆盖的方法,那么,父类f方法原本的实现将会被子类影响。但是这一点子类可能无法知道自己的覆盖会对父类的方法造成这些影响。
这是因为由于自用性的存在,我们必须十分了解父类的实现,而且,父类随时可能更改,子类的一些假设会被推翻。
什么是自用性?
就是父类里面的方法调用了其他父类里面的方法,这个现象在继承的时候会有问题。如果父类的一个方法f使用了一个可以被子类覆盖的方法,那么,父类f方法原本的实现将会被子类影响。但是这一点子类可能无法知道自己的覆盖会对父类的方法造成这些影响。
有一个例子:这里的f方法是一个通用方法,所以提到了父类,所有子类都可以使用,但是该方法是自用的,此时如果子类覆盖了overide方法,那么父类f就会使用子类的方法,父类f的行为也就改变了。那么为什么父类的f会调用子类的overide呢?子类对象的行为是在运行时绑定的,也就是调哪一个方法完全由调用者决定。我们创建的是a对象,是一个子类对象,a调用了f方法,此时发现子类没有该方法,就会调用父类的方法,然后f内部调用了overide方法,此时overide应该取子类的还是父类的?这个由调用对象决定,也就是this,相当于this.overide(),this也就是f的调用者,指向的是子类对象a。因为a是子类对象且覆盖了overide方法,所以此时优先调用子类的方法。通过这个例子我们知道,由于父类自用性可能的存在,一旦父类使用了子类覆盖的方法,那么就会影响到父类的方法。所以在继承的时候,我们必须了解父类的方法实现,这是一种破坏封装的行为。
public class Base {
public void f(){
overide();
}
public void overide(){
System.out.println("zilei");
}
}
public class Sub extends Base{
public void overide(){
System.out.println("zilei");
}
public static void main(String args[]){
Base a = new Sub();
a.f();
}
}
刚才的例子只是逻辑错误,有时候会导致运行时错误。那就是当这种自用性发生在了父类的构造方法里,如果父类的构造方法使用了被覆盖的方法,该方法调用了子类成员的方法,此时子类成员还没有被初始化,那么就会导致空指针。
public class Base {
Base (){
overide();
}
void overide(){
System.out.println("fulei");
}
}
public class Sub extends Base {
String s;
Sub(){
s = "123";
}
void overide(){
System.out.println(s.length());
}
public static void main(String args[]){
Base a = new Sub();
}
}
还有一点,父类的改动是可能的,这时一旦父类改动了,子类不知道,子类可能就会被影响。