继承适用于is-a关系,如Animal派生Dog;组合适用于has-a关系,如Car包含Engine。优先使用组合,避免过度继承导致耦合,提升可维护性与扩展性。

在Java面向对象编程中,组合与继承是两种实现代码复用的核心机制。它们各有适用场景,合理选择能提升代码的可维护性与扩展性。
继承:适用于“是什么”关系
继承表达的是is-a关系,即子类是父类的一种特殊形式。当多个类共享相同的属性和行为,并且逻辑上属于同一类型体系时,使用继承更自然。
常见应用场景包括:
- 定义通用基类,如Animal类,派生出Dog和Cat
- 框架设计中提供抽象模板,如Spring中自定义异常继承RuntimeException
- 需要多态支持时,如通过父类引用调用子类重写方法
注意:Java不支持多重继承(除接口外),过度使用继承可能导致类层次复杂、耦合度高。
立即学习“Java免费学习笔记(深入)”;
组合:适用于“有一个”关系
组合体现的是has-a关系,即一个类包含另一个类的实例作为成员变量。它强调功能的组装,而非类型继承。
典型使用情况有:
- 构建复杂对象,如Car类包含Engine、Wheel对象
- 实现策略模式,将算法封装为独立类,由上下文类持有其引用
- 替代多重继承,通过组合多个组件类来复用功能
组合的优势在于灵活性高,修改组件不影响主体结构,符合开闭原则。
实际项目中的选择建议
优先使用组合,除非明确存在类型继承关系。例如:
- 用户权限系统中,AdminUser不应继承User,而应通过角色组合实现权限叠加
- 图形绘制工具中,Shape类可组合Color和Border对象,便于动态更换样式
- 服务层组件常通过依赖注入组合DAO或Feign客户端,解耦业务逻辑与数据访问
基本上就这些,关键在于理清模型间的语义关系,避免为了复用而强行继承。










