模板方法模式
什么是模板方法
模板方法模式:定义一个操作中的算法骨架,而将一些步骤延迟到子类中。模板方法使得子类可以不改变一个算法的结构即可重定义该算法的
重复代码全部在父类里面,不同业务的,抽取给子类进行实现。抽取过程---抽象方法。
某些特定步骤。
核心:处理某个流程的代码已经都具备,但是其中某个节点的代码暂时不能确定。因此,我们采用工厂方法模式,将这个节点的代码实现转移给
子类完成。即:处理步骤在父类中定义好,具体的实现延迟到子类中定义。
说白了,就是将一些相同操作的代码,封装成一个算法的骨架。核心的部分留在子类中操作,在父类中只把那些骨架做好。
例如:
1.去银行办业务,银行给我们提供了一个模板就是:先取号,排对,办理业务(核心部分我们子类完成),给客服人员评分,完毕。
这里办理业务是属于子类来完成的,其他的取号,排队,评分则是一个模板。
2.去餐厅吃饭,餐厅给提供的一套模板就是:先点餐,等待,吃饭(核心部分我们子类完成),买单
这里吃饭是属于子类来完成的,其他的点餐,买单则是餐厅提供给我们客户的一个模板。
什么时候使用模板方法
实现一些操作时,整体步骤很固定,但是呢。就是其中一小部分容易变,这时候可以使用模板方法模式,将容易变的部分抽象出来,供子类实现。
中应用场景
其实,各个框架中,都有模板方法模式的影子。
数据库访问的封装、Junit单元测试、servlet中关于doGet/doPost方法的调用
Hibernate中模板程序、spring中JDBCTemplate,HibernateTemplate等等
模板方法具体实现
这里使用银行办理业务为例
首先,定义一个模板。模板中把办理业务用作核心部分,让子类来实现。
不同的业务抽取到子类去实现
首先建立一个模板出来:
先定义一个模板:
package TemplateModel;
public abstract class MsgTemplate {
public void sendMsg(){
//1.开始日志报文
addHeadLog();
//2.调用具体不同运营商发送
httpRequest();
//3.结束日志报文
addFootLog();
}
private void addFootLog() {
System.out.println("调用运营商开始记录日志...");
}
public abstract void httpRequest(); //调用不同的运营商 需要暂时不写死 定义为抽象的 同时 类也要定义为抽象的 而且必须public类型的
private void addHeadLog() {
System.out.println("调用运营商结束记录日志... ");
}
}
移动:
package TemplateModel;
public class YiDong extends MsgTemplate { //移动
@Override
public void httpRequest() {
System.out.println("移动运营商的接口");
}
}
联通:
package TemplateModel;
public class LianTong extends MsgTemplate{
@Override
public void httpRequest() {
System.out.println("联通运营商的接口");
}
}
客户端调用:
package TemplateModel;
public class Client {
public static void main(String[] args) {
MsgTemplate yidong = new YiDong();
yidong.sendMsg();
}
}
运行结果:
通过继承关系 继承
模板方法可以实现aop功能哈,aop做面向切面编程的,模板方法是把整体业务抽取出来形成一个模板。aop没有模板概念
模板方法整体是个固定的
Servlet中的doGet doPost都是模板方法模式的
要使用servlet 首先要继承HttpServlet: HttpServlet源码:
====>
HttpServlet是个抽象类,经验判断 在看源码时候 如果看到了抽象类 一般会是 模板方法模式 类里面的抽象方法 到子类中去重写
重写了doPOST方法 其他的业务进行综合集中处理了 图太多我就不给大家一一截图了 大家自己研究下~~
模板方法模式就是定义一个算法的骨架,而将具体的算法延迟到子类中实现
优点: 使用模板方法模式,在定义算法骨架的同时,可以很灵活的实现具体的算法,满足用户灵活多变的需求
缺点: 如果算法骨架有修改的话,则需要修改抽象类
package com.toov5.reflact;
abstract class getTime{
public final long getTime() { //不让子类去重写
long start = System.currentTimeMillis();
code();
long end = System.currentTimeMillis();
return end-start;
}
public abstract void code(); //让子类去实现 想怎么写就怎么写
}
//子类
class Demo1 extends getTime{
@Override
public void code() {
for(int i=0; i<10000; i++) {
System.out.println("不惧");
}
}
}
public class ReflectTest9 {
public static void main(String[] args) {
Demo1 demo1 = new Demo1();
System.err.println(demo1.getTime());
}
}