为了保证父类有良好的封装性,不会被子类随意修改,设计父类通常应该遵循以下规则:
1、尽量隐藏父类的内部数据,尽量把父类的所有成员变量设置为 private 访问类型,不要让子类直接访问父类的成员变量;
2、不要让子类可以随意访问和修改父类方法,父类中仅为辅助其他工具的方法,应该使用private 访问控制符修饰,让子类无法访问该方法;如果父类中的方法需要被外部类调用,则必须设置为 public 修饰,但不希望子类重写该方法,可以使用final修饰符来修饰该方法;如果希望父类的某个方法被子类重写,但不希望被其他类自由访问,则可以使用protected来修饰该方法。
3、尽量不要在父类构造器中调用将要被子类重写的方法。
packagetest_Java;classBase {publicBase()
{
test();
}public voidtest()//1
{
System.out.println("将被子类重写的方法");
}
}public classSub {privateString name;public voidtest()//2
{
System.out.println("子类重写父类的方法, "+"其name字符串的长度"+name.length());
}public static voidmain(String[] args) {//TODO Auto-generated method stub
Sub s = newSub();
s.test();//空指针异常
}
}
产生空指针异常的原因:当系统试图创建Sub对象时,同样会先执行其父类构造器,如果父类构造器调用了被其子类重写的方法,则变成调用被子类重写后的方法,当创建Sub对象时,会先执行Base类中的Base构造器,而Base构造器中调用test()方法-----并不是1而是2,此时的Sub对象的name实例变量是null,因此将引发空指针异常。
到底何时需要从父类派生出新的子类,具备以下条件:
子类需要额外增加属性,不仅仅是属性值的改变。
子类需要增加自己独有的行为方式,
如果只是出于类复用的目的,并不一定需要使用继承,完全可以使用组合来实现。
classAnimal{private voidbeat()
{
System.out.println("心脏跳动。。。");
}public voidbreath()
{
beat();
System.out.println("吸一口气,吐一口气,呼吸中。。。");
}
}classBird{privateAnimal a;publicBird(Animal a)
{this.a=a;
}public voidBreath()
{
a.breath();
}public voidfly()
{
System.out.println("i can fly.");
}
}public classCompositeTest {public static voidmain(String [] args)
{
Animal a= newAnimal();
Bird b= newBird(a);
b.Breath();
b.fly();
}
}
总之继承要表达的是一种“是(is a)”的关系,而组合表达的是“有(has a)”的关系。