真正的组合需同时满足:构造器中直接new创建、不提供public方法暴露引用、显式释放资源;否则仅为聚合或误用。
Java 中的组合(Composition)是一种强依赖的 has-a 关系:整体对象负责创建、持有并销毁部分对象,部分对象的生命周期完全绑定在整体上。比如 Car 持有 Engine 实例,Engine 不会单独 new 出来供其他类复用,也不会在 Car 销毁后继续存活。
不是所有成员变量赋值都叫组合。真正体现组合语义,需同时满足:
Engine 实例在 Car 的构造器中直接 new 出来(不通过参数传入或工厂方法延迟获取)Car 类不提供任何 public 方法返回该 Engine 引用(避免外部篡改或长期持有)Car 的 finalize() 或 close()(如有)应显式释放 Engine 相关资源(如关闭线程、释放 native 句柄等)public class Car {
private final Engine engine; // final 表明不可替换
public Car() {
this.engine = new Engine(); // 在构造器中创建 → 生命周期绑定
}
// ❌ 不暴露 engine 引用
// public Engine getEngine() { return engine; }
public void start() {
engine.ignite();
}
}
聚合(Aggregation)看起来也像 has-a,但部分对象可独立存在、被多个整体共享,甚至由外部管理生命周期。典型区别:
Order 包含 OrderItem → OrderItem 由 Order 构造器 new,且不出现在其他地方Department 包含 Employee 列表 → Employee 由 HR 系统创建,可能同时属于多个项目组private final 且只在构造器初始化,大概率是组合;如果是 setter 注入或集合 add 进来的,大概率是聚合实际编码中,开发者容易把“语法上有成员变量”等同于“语义上是组合”,结果破坏封装或引发内存泄漏:
DatabaseConnection 放进静态字段或单例池里 → 部分脱离了整体生命周期,变*局共享,不再是组合toString() 或日志中直接打印组合对象的引用(如 this.engine.toString()),却没重写 Engine.toString() → 导致无限递归或堆栈溢出@Data 自动生成 getter → 
组合不是语法糖,是设计契约:一旦声明某字段是组合,就要对它的创建、使用、销毁全程负责。漏掉任意一环,语义就垮了。