1. 概述
建造者(Builder)设计模式用于组装具有复杂结构的实例。在构建一个复杂实例的时候,我们很难一气呵成,我们首先把实例的各个部分建造出来,然后分阶段把它们组装起来。
建造者设计模式中主要有三个角色:抽象建造者,具体建造者,指挥者。
抽象建造者通常是一个抽象类,仅仅定义了方法,具体建造者继承抽象建造者,实现抽象建造者的中定义的方法,指挥者负责调用建造者去构建实例。
建造者设计模式的优点:
(1) 增加新的具体建造者无须修改原有代码,指挥者负责调用抽象建造者去构建实例,系统扩展方便,符合“开闭原则” 。
(2) 指挥者无需知道具体建造过程,指挥者负责调用抽象建造者去构建实例,使得相同的创建过程可以创建不同的产品对象。
(3) 将复杂的构建步骤分解在不同的方法中,使得构建过程更加清晰, 也更方便使用程序来控制构建过程。
建造者的缺点:
(1) 建造者模式所创建的实例一般具有较多的共同点,其组成部分相似,如果实例之间的差异性很大,则不适合使用。
(2) 如果所创建的实例实例的内部变化复杂,需要定义很多具体建造者类来实现这种变化,导致系统变得很庞大,开发和维护的难度会上升,因此在这种情况下,要考虑是否选择建造者模式。
2. 示例程序
一个使用Builder设计模式来编写“文档的程序”的例子,文档的结构有:一个标题、几个字符串、条目项目。
示例程序中有如下四个类:Builder类、Director类、TextBuilder类、HTMLBuilder类。
Builder类,抽象建造者,定义了编写文档的方法。Builder类是抽象类,仅仅是定义了方法,具体处理交给子类。
Director类,监工,使用该方法编写一个具体的文档,负责使用建造者的类来生成实例。
Builder类的两个子类TextBuilder类和HTMLBuilder类,两个具体的建造者,TextBuilder编写纯文本文档,HTMLBuilder类编写HTML文档。
Builder.java 定义了编写文档的方法,如下所示:
package builder;
public abstract class Builder {
public abstract void makeTitle(String title);
public abstract void makeString(String str);
public abstract void makeItems(String[] items);
public abstract void close();
}
Director.java,监工,负责使用建造者的类来生成实例:
package builder;
public class Director {
public Builder builder;
public Director(Builder builder) {
this.builder = builder;
}
// 编写文档
public void construct() {
builder.makeTitle("Hello Builder");
builder.makeString("从早上到下午");
builder.makeItems(new String[] {
"早上好",
"中午好",
"下午好"
});
builder.makeString("晚上");
builder.makeItems(new String[] {
"晚上好",
"晚安",
"再见"
});
builder.close();
}
}
TextBuilder.java,具体建造者类,getResult()返回建造的结果。
package builder;
public class TextBuilder extends Builder{
private StringBuffer buffer = new StringBuffer();
@Override
public void makeTitle(String title) {
buffer.append("==================================\n");
buffer.append("[" + title + "]\n");
buffer.append("\n");
}
@Override
public void makeString(String str) {
buffer.append('*' + str + "\n");
buffer.append("\n");
}
@Override
public void makeItems(String[] items) {
for (int i=0; i<items.length; i++) {
buffer.append(" ." + items[i] + "\n");
}
}
@Override
public void close() {
buffer.append("==================================\n");
}
public String getResult() {
return buffer.toString();
}
}
HTMLBuilder.java,具体建造者类,getResult()返回建造的结果。
package builder;
import java.io.FileWriter;
import java.io.IOException;
import java.io.PrintWriter;
public class HTMLBuilder extends Builder{
private String filename;
private PrintWriter writer;
@Override
public void makeTitle(String title) {
filename = title + ".html";
try {
writer = new PrintWriter(new FileWriter(filename));
} catch (IOException e) {
e.printStackTrace();
}
writer.println("<html><head><title>" + title + "</title></head></html>");
writer.println(title);
}
@Override
public void makeString(String str) {
writer.println("<p>" + str + "</p>");
}
@Override
public void makeItems(String[] items) {
writer.println("<ul>");
for (int i=0; i<items.length; i++) {
writer.println("<li>" + items[i] + "</li>");
}
writer.println("</ul>");
}
@Override
public void close() {
writer.println("</body></html>");
writer.close();
}
public String getResult() {
return filename;
}
}
使用者类,User.java。
package builder;
public class User {
public static void main(String args[]) {
TextBuilder textBuilder = new TextBuilder();
Director director = new Director(textBuilder);
director.construct();
String result = textBuilder.getResult();
System.out.println(result);
HTMLBuilder htmlBuilder = new HTMLBuilder();
Director director01 = new Director(htmlBuilder);
director01.construct();
}
}
运行截图如下所示:
参考文献:
- 图解设计模式 - 结成浩著、杨文轩译 - 人民邮电出版社
- 尚硅谷 - 图解设计模式