模板方法设计模式
模式是一个解决方案,一个模式解决了一类特定的问题,当我们再次遇到同样的问题时,我们仍然可以使用它解决同样的问题,这里介绍使用模板方法设计模式。
在开发中我们时常可以使用模板方法设计模式,这里我们就以简单的上班为例。上班一族每个工作日去上班都要经过 洗漱---上班方式(交通工具)--开始工作 三个步骤,我们写一个程序来模拟这个过程
public class Work { public void doWork() { //洗漱过程 System.out.println("洗漱。。。。。。"); //去公司过程 System.out.println("上班方式:地铁。。。。。。"); //工作过程 System.out.println("开始工作。。。。。。"); } }
然后有些人上班是走路,骑自行车,公交等等方式去上班,但不论选择何种交替工具,都得先洗漱,最后工作。当需要公交上班时,很容易想到由于只是交通工具不同,我们可以把Work类快速复制一份,只需要把
公交换成公交,如:
public void doWork() { //洗漱过程 System.out.println("洗漱。。。。。。"); //去公司过程 System.out.println("上班方式:公交。。。。。。"); //工作过程 System.out.println("开始工作。。。。。。"); }
复制黏贴看起来很实用,但是这些类会变得难以维护,如果公共部分洗漱需要更改,这需要把类一个个进行更改,所以我们需要好好思考复制黏贴真的好用吗?
我们都知道软件开发,变化时永恒的,软件维护始终贯穿软件开发整个过程。为了提高软件的维性,我们在软件开发中应该遵守DRY这一重要原则。
DRY(Don‘t Repeat Yourself)不要复制自己,为了使软件更加健壮,易于阅读和维护,应遵守该原则,不要编写重复的代码,给维护代理麻烦。
上面的情形,我们可以使用模板方法设计模式来解决。
1.构建Work抽象父类
public abstract class Work { //定义final,不被子类重写 public final void clean() { //洗漱过程 System.out.println("洗漱。。。。。。"); } //去上班方式 public abstract void goCompany(); public final void work() { //工作过程 System.out.println("开始工作。。。。。。"); } }
2.子类继承抽象父类,实现自己的上班方式
public class Subway extends Work{ @Override public void goCompany() { //去公司过程 System.out.println("上班方式:地铁。。。。。。"); } }
使用模板方法设计模式能够解决代码冗余,易于扩展。
模板方法设计模式应用很广泛,但是过分的使用模板设计模式往往会导致类爆炸(子类泛滥)。如查询数据库数据并对查询结果进行处理。整个过程可以分为三步
1.创建数据库连接对象
2.创建Statement实例执行查询操作
3.处理查询结果
发现整个过程中1,2步是相同的,不同的主要是对查询结果进行操作上。这里使用模板方法设计模式你会发现,由于查询的种类很多,导致需要创建很多子类来处理查询结果,会导致子类泛滥。为了解决这一问题,需要结合接口回调来处理,使用java的匿名内部类来完成查询结果的处理。
1.定义父类
package com.longpo.template; public class Work { public final void Connect() { System.out.println("连接数据库。。。。。。"); } public void statement() { System.out.println("Statement实例。。。。。。"); } public void handler(CallBack callBack){ this.Connect(); this.statement(); callBack.HanderResult("处理查询结果。。。。。。"); } //回调接口 public interface CallBack { public void HanderResult(String result); } }
2.测试
public class Person extends Work{ public static void main(String[] args) { Person person =new Person(); person.handler(new CallBack() { @Override public void HanderResult(String result) { System.out.println(result); } }); } }
运行结果: