模mu
版模式
模板方法模式,一般是为了统一子类的算法实现步骤,所使用的一种手段或者说是方式。它在父类中定义一系列算法的步骤,而将具体的实现都推迟到子。最典型的形式就是一个接口,一个抽象父类,父类中会有一系列的抽象方法,而在子类中去一一实现这些方法。
模板方法模式并不强制接口的实现类必须继承,所以不会对子类造成任何影响,而如果子类的实现可以配得上模板类的模板,那么就可以享受模板方法模式带来的。通常情况下,模板方法模式用于--定义构建某个对象的步骤与顺序,或者定义一个算法的骨架。
实例
所有的依赖最好是依赖抽象,易扩展。
用于生成网页页面的类:
public interface PageBuilder {
String bulidHtml();
}
这个接口很简单,就是直接制造一个Html页面的内容,假设我们不使用模板方法模式,直接让各个子类去直接实现这个接口,那么肯定实现的方式千奇百怪,而且步骤也乱七八糟的,这样实在不利于维护和扩展。所以我们可以使用模板方法模式,将这个过程给制定好,然后把具体的内容填充交给子类就好,这样这些子类生成的HTML页面就会非常统一。
基于这个目的,我们定义如下抽象类,去实现这个接口,并且我们定义好步骤。
具体实现生成html算法的类,(包含关键步骤及一些自定义交给子类实现的方法)
public abstract class AbstractPageBuilder implements PageBuilder{
private StringBuffer stringBuffer = new StringBuffer();
public String bulidHtml() {
//首先加入doctype,因为都是html页面,所以我们父类不需要推迟给子类实现,直接在父类实现
stringBuffer.append("<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Transitional//EN\" \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd\">");
//页面下面就是成对的一个HTML标签,我们也在父类加入,不需要给子类实现
stringBuffer.append("<html xmlns=\"http://www.w3.org/1999/xhtml\">");
//下面就应该是head标签里的内容了,这个我们父类做不了主了,推迟到子类实现,所以我们定义一个抽象方法,让子类必须实现
appendHead(stringBuffer);
//下面是body的内容了,我们父类依然无法做主,仍然推迟到子类实现
appendBody(stringBuffer);
//html标签的关闭
stringBuffer.append("</html>");
return stringBuffer.toString();
}
//第一个模板方法
protected abstract void appendHead(StringBuffer stringBuffer);
//第二个模板方法
protected abstract void appendBody(StringBuffer stringBuffer);
}
实现类:
public class MyPageBuilder extends AbstractPageBuilder{
protected void appendMeta(StringBuffer stringBuffer) {
stringBuffer.append("<meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\" />");
}
protected void appendTitle(StringBuffer stringBuffer) {
stringBuffer.append("<title>你好</title>");
}
protected void appendBody(StringBuffer stringBuffer) {
stringBuffer.append("<body>你好,世界!</body>");
}
public static void main(String[] args) {
PageBuilder pageBuilder = new MyPageBuilder();
System.out.println(pageBuilder.bulidHtml());
}
}
模板方法模式还有一种使用的方式,为了给子类足够的自由度,可以提供一些方法供子类覆盖,去实现一些骨架中不是必须但却可以有自定义实现的步骤,类似于缺省适配模式。
包含缺省实现的模板类
public abstract class AbstractPageBuilder implements PageBuilder{
private static final String DEFAULT_DOCTYPE = "<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Transitional//EN\" \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd\">";
private static final String DEFAULT_XMLNS = "http://www.w3.org/1999/xhtml";
private StringBuffer stringBuffer = new StringBuffer();
public String bulidHtml() {
stringBuffer.append(DEFAULT_DOCTYPE);
stringBuffer.append("<html xmlns=\"" + DEFAULT_XMLNS + "\">");
stringBuffer.append("<head>");
appendTitle(stringBuffer);
appendMeta(stringBuffer);
appendLink(stringBuffer);
appendScript(stringBuffer);
stringBuffer.append("</head>");
appendBody(stringBuffer);
stringBuffer.append("</html>");
return stringBuffer.toString();
}
protected void appendMeta(StringBuffer stringBuffer){
}
protected void appendLink(StringBuffer stringBuffer){
}
protected void appendScript(StringBuffer stringBuffer){
}
protected abstract void appendTitle(StringBuffer stringBuffer);
protected abstract void appendBody(StringBuffer stringBuffer);
}