介绍:模板模式定义了一个模板抽象类,这个抽象类中定义了方法调用的形式,顺序。子类通过重写对方法进行实现,但是调用方式不能改变。
模板模式中的模板中定义了核心的代码骨架,一些有着不同方式实现的代码放在子类中,模板模式规定了方法执行的方式或者可以说是方法的执行顺序,执行结构,使得子类只能遵守这些规定。
模板模式解决了很多代码在子类中通用。每次业务相同的代码,用户每次还要重写这些一样的代码,现在将这些通用的代码提取出来,放在一个抽象类中,在抽象类中规定方法执行的方式,一些具体实现不一样的方法定义为抽象方法,由子类通过继承抽象类实现这些抽象方法。也可以说是把通用的算法抽象出来。
把固定的部分封装起来,对于可变的部分用户可以进行扩展,提取出代码的公共部分放在父类,行为由父类控制,子类负责实现可变的部分。
在tomcat中就有模板模式的应用,在用户写的Servlet中的doget和dopost方法就是子类中重写的方法,HttpServlet中的service方法定义了方法执行的方式,以下是HttpServlet中service方法的源码,方法中定义了每个方法执行的顺序,方式。也就是规定了核心,用户需要对那些服务定义自己的代码。
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { String method = req.getMethod(); long lastModified; if (method.equals("GET")) { lastModified = this.getLastModified(req); if (lastModified == -1L) { this.doGet(req, resp); } else { long ifModifiedSince; try { ifModifiedSince = req.getDateHeader("If-Modified-Since"); } catch (IllegalArgumentException var9) { ifModifiedSince = -1L; } if (ifModifiedSince < lastModified / 1000L * 1000L) { this.maybeSetLastModified(resp, lastModified); this.doGet(req, resp); } else { resp.setStatus(304); } } } else if (method.equals("HEAD")) { lastModified = this.getLastModified(req); this.maybeSetLastModified(resp, lastModified); this.doHead(req, resp); } else if (method.equals("POST")) { this.doPost(req, resp); } else if (method.equals("PUT")) { this.doPut(req, resp); } else if (method.equals("DELETE")) { this.doDelete(req, resp); } else if (method.equals("OPTIONS")) { this.doOptions(req, resp); } else if (method.equals("TRACE")) { this.doTrace(req, resp); } else { String errMsg = lStrings.getString("http.method_not_implemented"); Object[] errArgs = new Object[]{method}; errMsg = MessageFormat.format(errMsg, errArgs); resp.sendError(501, errMsg); } }
假设我们去了饭店,干什么,当然是去吃饭喽,我们到饭店吃饭得有个流程吧,首先我们到了饭店,饭店需要为你找座位吧,当你坐下之后,服务员给你一张菜单,让你点菜,这都是固定的顺序,但是你点什么菜不是饭店规定好的,可以随意点,点完之后服务员会把点的这些菜提交给后厨,后厨做好之后服务员端上来,吃完之后就要到前台结账买单。除了顾客点菜之外,剩下的步骤都是一套固定的流程,我们就可以定义一个模板类,规定好饭店的这些流程,只有点餐这个流程不需要具体规定,留给顾客自己选择。
定义模板类:饭店中时规定了方法执行的流程和具体操作,但是顾客点菜是顾客自己的行为,不需要规定
package demo_template; public abstract class Template { //服务员为顾客选择座位 private void selectSeat() { System.out.println("选择一个座位"); } //服务员给顾客菜单 private void passMenu() { System.out.println("递菜单"); } //假设顾客只点了A,B,C三个菜 public abstract void selectA(); public abstract void selectB(); public abstract void selectC(); //服务员提交菜单 private void submitMenu() { System.out.println("提交菜单"); } //上菜 private void serving() { System.out.println("上菜"); } //顾客买单 private void pay() { System.out.println("顾客买单"); } protected void service() { selectSeat(); passMenu(); this.selectA(); this.selectB(); this.selectC(); submitMenu(); serving(); pay(); } }
定义第一个顾客:顾客只负责点菜,其他不需要关心
package demo_template; public class First extends Template { @Override public void selectA() { System.out.println("地三鲜"); } @Override public void selectB() { System.out.println("大米饭"); } @Override public void selectC() { System.out.println("橙汁"); } }
第二位顾客:
package demo_template; public class Second extends Template { @Override public void selectA() { System.out.println("大米饭"); } @Override public void selectB() { System.out.println("糖醋鲤鱼"); } @Override public void selectC() { System.out.println("地三鲜"); } }
测试类:
package demo_template; public class Test { public static void main(String[] args) { System.out.println("第一位顾客:"); Template first = new First(); first.service(); System.out.println("---------------------"); System.out.println("第二位顾客:"); Template second = new Second(); second.service(); } }
测试类用到了多态,我们创建的对象是First和Second的,类型是Template的,模板类中的三个点菜方法是用的this调用,this代表当前对象,当前对象是谁的,当然是First和Second的。
输出:
第一位顾客: 选择一个座位 递菜单 地三鲜 大米饭 橙汁 提交菜单 上菜 顾客买单 --------------------- 第二位顾客: 选择一个座位 递菜单 大米饭 糖醋鲤鱼 地三鲜 提交菜单 上菜 顾客买单
优点:
1、封装不变部分,扩展可变部分。
2、提取公共代码,便于维护。
3、行为由父类控制,子类实现。
缺点:每一个不同的实现都需要一个子类来实现,导致类的个数增加,使得系统更加庞大。
使用场景:
1、有多个子类共有的方法,且逻辑相同。
2、重要的、复杂的方法,可以考虑作为模板方法。
参考:
http://blog.csdn.net/lovelion/article/details/8299794
http://www.runoob.com/design-pattern/template-pattern.html