Builder 模式简介
Builder 模式主要用来组装复杂的实例。
示例程序
示例程序类图
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类
public class Director {
private Builder builder;
public Director(Builder builder){
this.builder = builder;
}
/**
* 编写文档
*/
public void construct(){
// 标题
builder.makeTitle("Greeting");
// 字符串
builder.makeString("从早上至下午");
// 条目
builder.makeItems(new String[]{"早上好。","下午好。"});
// 其他字符串
builder.makeString("晚上");
// 其他条目
builder.makeItems(new String[]{"晚上好","晚安","再见"});
// 完成文档
builder.close();
}
}
TextBuilder类
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"); // 为条目添加・
}
buffer.append("\n"); // 换行
}
@Override
public void close() { // 完成文档
buffer.append("==============================\n"); // 装饰线
}
public String getResult() { // 完成的文档
return buffer.toString(); // 将StringBuffer变换为String
}
}
HTMLBuilder
public class HTMLBuilder extends Builder {
private String filename; // 文件名
private PrintWriter writer; // 用于编写文件的PrintWriter
@Override
public void makeTitle(String title) { // HTML文件的标题
filename = title + ".html"; // 将标题作为文件名
try {
writer = new PrintWriter(new FileWriter(filename)); // 生成 PrintWriter
} catch (IOException e) {
e.printStackTrace();
}
writer.println("<html><head><title>" + title + "</title></head><body>"); // 输出标题
writer.println("<h1>" + title + "</h1>");
}
@Override
public void makeString(String str) { // HTML文件中的字符串
writer.println("<p>" + str + "</p>"); // 用<p>标签输出
}
@Override
public void makeItems(String[] items) { // HTML文件中的条目
writer.println("<ul>"); // 用<ul>和<li>输出
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; // 返回文件名
}
}
Main类
public class Main {
public static void main(String[] args) {
if (args.length != 1) {
usage();
System.exit(0);
}
if (args[0].equals("plain")) {
TextBuilder textbuilder = new TextBuilder();
Director director = new Director(textbuilder);
director.construct();
String result = textbuilder.getResult();
System.out.println(result);
} else if (args[0].equals("html")) {
HTMLBuilder htmlbuilder = new HTMLBuilder();
Director director = new Director(htmlbuilder);
director.construct();
String filename = htmlbuilder.getResult();
System.out.println(filename + "文件编写完成。");
} else {
usage();
System.exit(0);
}
}
public static void usage() {
System.out.println("Usage: java Main plain 编写纯文本文档");
System.out.println("Usage: java Main html 编写HTML文档");
}
}
Builder 模式中的登场角色
1.Builder(建造者)
Builder角色负责定义用于生成实例的接口(API)。Builder角色中准备了用于生成实例的方法。在实例程序中,由Builder类扮演此角色。
2.ConcreteBuilder(具体的建造者)
ConcreteBuilder角色是负责实现Builder角色的接口的类(API)。这里定义了在生成实例时实际被调用的方法。此外,在ConcreteBuilder角色中还定义了获取最终生成结果的方法。在示例程序中,由TextBuilder类和HTMLBuilder类扮演此角色。
3.Director(监工)
Director角色负责使用Builder角色的接口(API)来生成实例。它并不依赖于ConcreteBuilder角色。为了确保不论ConcreteBuilder角色是如何被定义的,Director角色都能正常工作,它只调用在Builder角色中定义的方法。在示例程序中,由Director类扮演此角色。
4.Client(使用者)
该角色使用了Builder模式。在示例程序中,由Main类扮演此角色。