java中的继承与组合
(一)使用继承的注意点
继承带来了高度的复用性的同时,也带来了一个问题:继承严重破坏了父类的封装性!!我们在学习封装的时候知道:每个类都应该封装它的内部信息和实现细节,而只暴露必要的方法给其他的类使用。但是在继承关系中,子类可以直接访问父类的成员变量和方法,从而造成子类和父类的严重耦合。
为了保证父类有良好的封装性,不会被子类随意改变,设计父类时我们应当遵循以下的原则:
1. 尽量隐藏父类的内部数据,尽量把父类的所有成员变量都设置成private访问类型。
2. 不要让子类随意访问和修改父类的方法。父类中那些仅为辅助其他的工具方法,应该使用private访问控制符来修饰,让子类无法访问;如果必须要使用public,但是又不希望子类重写该方法,可以使用final修饰符;如果希望父类的某个方法能够被子类重写,但是不希望被其他类自由访问的话,可以用protected来修饰该方法
3. 尽量不要在父类构造器中调用将要被子类重写的方法。
class Animal
{
private void beat()
{
System.out.println("心脏跳动...");
}
public void breath()
{
beat();
System.out.println("吸一口气,吐一口气,呼吸中...");
}
}
// 继承Animal,直接复用父类的breath()方法
class Bird extends Animal
{
public void fly()
{
System.out.println("我在天空自在的飞翔...");
}
}
// 继承Animal,直接复用父类的breath()方法
class Wolf extends Animal
{
public void run()
{
System.out.println("我在陆地上的快速奔跑...");
}
}
public class InheritTest
{
public static void main(String[] args)
{
Bird b = new Bird();
b.breath();
b.fly();
Wolf w = new Wolf();
w.breath();
w.run();
}
}
在上面的代码中我们的wolf和bird两个类都继承了animal,并得到了其public的成员。
(二)利用组合实现复用
事实上,除了继承,我们还可以利用组合来实现复用,把旧有的类当成另一个类的组成部分,从而允许新类直接复用该类的public方法。
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;
}
// 重新定义一个自己的breath()方法
public void breath()
{
// 直接复用Animal提供的breath()方法来实现Bird的breath()方法。
a.breath();
}
public void fly()
{
System.out.println("我在天空自在的飞翔...");
}
}
class Wolf
{
// 将原来的父类组合到原来的子类,作为子类的一个组合成分
private Animal a;
public Wolf(Animal a)
{
this.a = a;
}
// 重新定义一个自己的breath()方法
public void breath()
{
// 直接复用Animal提供的breath()方法来实现Wolf的breath()方法。
a.breath();
}
public void run()
{
System.out.println("我在陆地上的快速奔跑...");
}
}
public class CompositeTest
{
public static void main(String[] args)
{
// 此时需要显式创建被组合的对象
Animal a1 = new Animal();
Bird b = new Bird(a1);
b.breath();
b.fly();
// 此时需要显式创建被组合的对象
Animal a2 = new Animal();
Wolf w = new Wolf(a2);
w.breath();
w.run();
}
}
到底是使用继承好还是组合好,我们应该结合具体情况和现实意义来考虑。