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