建造者模式:分步构建复杂对象的设计模式
建造者模式:分步构建复杂对象的设计模式
一、模式核心:分离对象构建与表示,支持复杂对象分步创建
在软件开发中,当创建一个复杂对象(如配置对象、组合对象)需要多个步骤(如属性设置、子对象初始化)时,直接通过构造方法创建会导致参数列表冗长、代码可读性差。
建造者模式(Builder Pattern) 将一个复杂对象的构建过程与其表示分离,允许使用相同的构建过程创建不同的表示。核心解决:
- 简化复杂对象创建:通过分步方法(如
setName()
、setAge()
)逐步构建对象,避免一次性传递所有参数。 - 支持链式调用:通过返回建造者自身实例,实现方法链式调用,提升代码流畅性。
- 解耦构建逻辑:将构建逻辑封装在建造者类中,便于扩展不同的构建策略(如普通用户与 VIP 用户的创建流程差异)。
核心思想与 UML 类图
建造者模式包含以下角色:
- 产品类(Product):需要构建的复杂对象,包含多个属性。
- 抽象建造者(Builder):定义构建产品各部分的接口,返回建造者实例(链式调用)。
- 具体建造者(Concrete Builder):实现抽象建造者接口,完成具体属性的设置。
- 指挥者(Director):控制构建流程,调用建造者的方法构建产品(可选角色)。
二、核心实现:用户对象的分步构建
1. 定义产品类(用户)
public class User { private String name; // 必选属性 private int age; // 可选属性 private String email; // 可选属性 private String address; // 可选属性 // 私有化构造方法,强制通过建造者创建 private User(Builder builder) { this.name = builder.name; this.age = builder.age; this.email = builder.email; this.address = builder.address; } // Getter 方法(省略 Setter,保证不可变性) public String getName() { return name; } public int getAge() { return age; } public String getEmail() { return email; } public String getAddress() { return address; } // 展示用户信息 public void show() { System.out.println("用户信息:"); System.out.println("姓名:" + name); System.out.println("年龄:" + age); System.out.println("邮箱:" + email); System.out.println("地址:" + address); }
}
2. 定义抽象建造者(内部类实现)
public class UserBuilder { private final String name; // 必选属性(构造方法传入) private int age; // 可选属性(默认值) private String email; // 可选属性(默认值 null) private String address; // 可选属性(默认值 null) // 构造方法强制设置必选属性 public UserBuilder(String name) { this.name = name; } // 可选属性设置方法,返回建造者自身(链式调用) public UserBuilder setAge(int age) { this.age = age; return this; } public UserBuilder setEmail(String email) { this.email = email; return this; } public UserBuilder setAddress(String address) { this.address = address; return this; } // 构建产品实例 public User build() { return new User(this); // 将建造者传递给产品构造方法 }
}
3. 客户端使用建造者模式
public class ClientDemo { public static void main(String[] args) { // 构建普通用户(设置部分属性) User normalUser = new UserBuilder("Alice") .setAge(25) .build(); // 构建高级用户(设置全部属性) User advancedUser = new UserBuilder("Bob") .setAge(30) .setEmail("bob@example.com") .setAddress("123 Main St") .build(); normalUser.show(); System.out.println("\n------------------------\n"); advancedUser.show(); }
}
输出结果:
用户信息:
姓名:Alice
年龄:25
邮箱:null
地址:null ------------------------ 用户信息:
姓名:Bob
年龄:30
邮箱:bob@example.com
地址:123 Main St
三、进阶:使用指挥者控制构建流程
当构建流程固定时,可以通过指挥者类封装构建步骤,客户端无需关心具体细节。
1. 定义指挥者类
public class UserDirector { private final UserBuilder builder; public UserDirector(UserBuilder builder) { this.builder = builder; } // 定义默认构建流程(如创建必填属性+年龄的用户) public User constructBasicUser() { return builder .setAge(18) // 默认年龄 18 岁 .build(); } // 定义高级构建流程(设置全部属性) public User constructFullUser(String email, String address) { return builder .setAge(18) .setEmail(email) .setAddress(address) .build(); }
}
2. 客户端通过指挥者构建对象
public class ClientDemo { public static void main(String[] args) { UserBuilder builder = new UserBuilder("Charlie"); UserDirector director = new UserDirector(builder); // 使用指挥者创建基础用户 User basicUser = director.constructBasicUser(); basicUser.show(); }
}
四、框架与源码中的建造者实践
1. JDK 中的 StringBuilder
StringBuilder
使用建造者模式实现字符串的分步拼接,支持链式调用:
String result = new StringBuilder("Hello") .append(" ") .append("World") .toString(); // 输出 "Hello World"
2. MyBatis 的 SqlSessionFactoryBuilder
MyBatis 通过建造者模式构建 SqlSessionFactory
,允许配置数据源、事务管理器等参数:
SqlSessionFactory factory = new SqlSessionFactoryBuilder() .build(Resources.getResourceAsStream("mybatis-config.xml"));
3. Android 的 AlertDialog.Builder
Android 中通过建造者模式创建对话框,分步设置标题、内容、按钮等:
new AlertDialog.Builder(context) .setTitle("提示") .setMessage("确认操作?") .setPositiveButton("确定", null) .create() .show();
五、避坑指南:正确使用建造者模式的 3 个要点
1. 区分建造者与工厂模式
- 工厂模式:专注于 “创建对象”,不关心构建过程(如
new User("Alice", 25)
)。 - 建造者模式:强调 “分步构建”,支持复杂对象的属性逐个设置(如
setName().setAge()
)。
2. 合理设计必选与可选属性
- 必选属性通过建造者的构造方法强制设置,避免出现不完整的产品实例。
- 可选属性通过公共方法设置,允许客户端按需选择。
3. 考虑线程安全
若建造者用于多线程环境,需对共享的产品属性或建造者实例添加同步控制(如 synchronized
)。
六、总结:何时该用建造者模式?
适用场景 | 核心特征 | 典型案例 |
---|---|---|
复杂对象分步构建 | 对象包含多个属性,需分步骤初始化 | 用户配置、订单详情、XML/JSON 解析 |
链式调用需求 | 需要通过方法链提升代码可读性 | 日志构建、SQL 语句拼接 |
构建流程多样化 | 相同构建步骤可生成不同表示的产品 | 报表生成(HTML/Excel/PDF 格式) |
建造者模式通过将复杂对象的构建过程分解为多个简单步骤,使代码更易读、可维护。下一篇我们将探讨桥接模式,解析如何分离抽象与实现,敬请期待!
扩展思考:建造者模式的变种
- 流式建造者:所有设置方法均返回
this
,形成方法链(如示例中的setAge().setEmail()
)。 - 建造者返回不同产品:一个建造者类支持构建多种相关产品(如用户与管理员对象)。