组合与聚合均实现“has-a”关系,区别在于生命周期管理:组合中部分随整体创建销毁,如Car与Engine;聚合中部分可独立存在,如School与Teacher,通过外部传入对象实现共享。

在Java中,类的组合与聚合是面向对象设计中实现“has-a”关系的重要方式。它们都表示一个类包含另一个类的对象作为成员变量,但语义和生命周期管理上有所不同。下面分别介绍如何实现以及两者的区别。
什么是组合与聚合
组合(Composition) 表示强拥有关系,整体与部分同生共死。比如“汽车”和“发动机”,发动机不能脱离汽车独立存在。
聚合(Aggregation) 是弱拥有关系,部分可以独立于整体存在。比如“学校”和“教师”,教师可以属于某个学校,也可以更换或独立存在。
如何实现组合
组合通过在类内部直接创建被包含对象的实例来实现,生命周期由容器类控制。
立即学习“Java免费学习笔记(深入)”;
- 成员变量声明为另一个类的类型
- 在构造函数中创建该对象实例
- 不接受外部传入该对象(或仅限内部使用)
示例:Car 和 Engine 的组合关系
class Engine {
public void start() {
System.out.println("引擎启动");
}
}
class Car {
private Engine engine; // 强关联
public Car() {
this.engine = new Engine(); // 内部创建,无法独立存在
}
public void start() {
engine.start();
System.out.println("汽车启动");
}
}
这里 Engine 是 Car 的一部分,Car 销毁时 Engine 也随之销毁。
在现实生活中的购物过程,购物者需要先到商场,找到指定的产品柜台下,查看产品实体以及标价信息,如果产品合适,就将该产品放到购物车中,到收款处付款结算。电子商务网站通过虚拟网页的形式在计算机上摸拟了整个过程,首先电子商务设计人员将产品信息分类显示在网页上,用户查看网页上的产品信息,当用户看到了中意的产品后,可以将该产品添加到购物车,最后使用网上支付工具进行结算,而货物将由公司通过快递等方式发送给购物者
如何实现聚合
聚合通过将已存在的对象引用传递给类来实现,被包含对象可被多个类共享。
- 成员变量仍是另一个类的引用
- 通过构造函数或 setter 方法传入对象
- 对象由外部创建,生命周期独立
示例:School 和 Teacher 的聚合关系
class Teacher {
private String name;
public Teacher(String name) {
this.name = name;
}
public String getName() {
return name;
}
}
class School {
private Teacher teacher;
public School(Teacher teacher) {
this.teacher = teacher; // 外部传入,可复用
}
public void showTeacher() {
System.out.println("本校教师:" + teacher.getName());
}
}
Teacher 可以先创建,再分配给 School,也可以被其他 School 使用或单独存在。
关键区别总结
理解组合与聚合的关键在于生命周期依赖和所有权:
- 组合:部分不能脱离整体,整体控制部分的创建与销毁
- 聚合:部分可独立存在,整体只是引用它
- 代码上区别主要体现在对象的创建时机和来源
基本上就这些。合理使用组合与聚合,可以让类的设计更符合现实逻辑,提升代码的可维护性和扩展性。









