组合优于继承,因灵活性高、耦合低;继承体现“is-a”关系,适用于多态和稳定基类;组合体现“has-a”关系,支持动态替换与更好维护。

在Java中,组合与继承都是实现代码复用的重要手段,但它们的设计思想和使用场景有明显区别。
继承:表示“是一个”的关系
继承通过extends关键字实现,子类继承父类的属性和方法,体现的是“is-a”关系。例如,狗是一个动物,就可以让Dog类继承Animal类。
优点:
- 可以直接复用父类的代码
- 支持多态,便于扩展程序行为
缺点:
立即学习“Java免费学习笔记(深入)”;
- 子类与父类高度耦合,父类修改可能影响所有子类
- 不加限制地使用会导致类层次复杂,难以维护
组合:表示“有一个”的关系
组合是将一个类的实例作为另一个类的成员变量,体现的是“has-a”关系。例如,汽车有一个引擎,就可以在Car类中包含一个Engine对象。
优点:
- 灵活性更高,可以在运行时动态替换组件
- 降低类之间的耦合度,更容易维护和测试
- 避免了继承带来的层级膨胀问题
例子:
<font face="monospace"> class Engine { void start() { System.out.println("引擎启动"); } } class Car { private Engine engine = new Engine(); // 组合 void start() { engine.start(); // 委托给引擎 System.out.println("汽车启动"); } } </font>
优先使用组合而非继承
Java设计原则中推荐优先使用组合而不是继承,因为组合更灵活、更安全。继承容易破坏封装性,而组合通过接口或委托来协作,能更好地应对变化。
什么时候用继承?
- 明确存在“is-a”关系
- 需要多态行为(如重写方法)
- 父类是受控的、稳定的基类
什么时候用组合?
- 类之间是“has-a”关系
- 希望在运行时改变行为(比如策略模式)
- 避免继承带来的紧耦合问题
基本上就这些。理解两者的关系和适用场景,有助于写出更清晰、可维护的Java代码。


