
在Java开发中,“组合优于继承”不是一句空话,而是提升代码可维护性与扩展性的关键设计原则。落地这个理念,核心是用“has-a”替代“is-a”,避免因继承导致的紧耦合和脆弱基类问题。以下通过实际项目场景说明如何将这一原则真正用起来。
用接口+组合替代多层继承
项目中常遇到需要复用行为的情况。比如订单系统中,普通订单、会员订单、团购订单都有“计算总价”的逻辑,但规则不同。若使用继承:
public class Order { ... }public class MemberOrder extends Order { ... }
public class GroupOrder extends Order { ... }
一旦出现“会员团购订单”,单继承就无法表达多重身份,且父类修改影响所有子类。
改用组合:
立即学习“Java免费学习笔记(深入)”;
public interface PricingStrategy {double calculatePrice(OrderContext context);
}
public class RegularPricing implements PricingStrategy { ... }
public class MemberPricing implements PricingStrategy { ... }
public class GroupPricing implements PricingStrategy { ... }
订单类持有策略对象:
public class Order {private PricingStrategy pricingStrategy;
public void setPricingStrategy(PricingStrategy strategy) {
this.pricingStrategy = strategy;
}
public double getTotal() {
return pricingStrategy.calculatePrice(this.context);
}
}
运行时动态注入策略,无需继承,行为更灵活。
封装共用能力为组件,而非抽象父类
多个服务需要日志记录、权限校验、通知发送等功能。传统做法是写一个 BaseService,其他服务继承它。
问题在于:子类被迫继承所有方法,哪怕用不到;BaseService 一改,全项目受影响。
更好的方式是把通用能力拆成独立组件:
public void send(String to, String msg) { ... }
}
public class AuditLogger {
public void log(String action) { ... }
}
业务服务通过字段引用这些组件:
public class OrderService {private NotificationService notifier;
private AuditLogger auditLogger;
public OrderService(NotificationService n, AuditLogger a) {
this.notifier = n;
this.auditLogger = a;
}
public void placeOrder(Order order) {
auditLogger.log("order placed");
// ...业务逻辑
notifier.send(order.getUserEmail(), "下单成功");
}
}
依赖通过构造函数注入,清晰可控,测试也更容易 mock。
避免“为了复用”而继承
常见误区:发现两个类有相同字段或方法,就提取父类。比如 User 和 Admin 都有 name、email,于是建 BaseUser。
但随着演化,User 可能加 address,Admin 加 roleLevel,父类越来越臃肿,子类也被迫承担无关字段。
更合理的做法是提取共用数据结构:
public class ContactInfo {private String name;
private String email;
}
User 和 Admin 内部包含 ContactInfo:
public class User {private ContactInfo contact;
private Address address;
}
这样变化隔离,也能实现代码复用,且不影响各自演进。
基本上就这些。组合让类职责更单一,依赖更明确,系统更容易应对需求变化。关键是转变思维:不要一上来就想“XX是不是一种YY”,而是问“XX有没有YYY能力”。










