书城计算机网络大话设计模式
8183300000157

第157章 9 模板方法模式与勾子方法(hookMethod)

大B:“刚才说模板方法模式运用于一个业务对象。事实上,框架频繁使用模板方法模式,使得框架实现对关键逻辑的集中控制。”

大B:“我们需要为基本Spring的应用做一个测试用例的基类。用于对类的方法进行单元测试。我们知道Spring应用把需要用到的对象都定义在外部的xml文件中,也称为context。”

大B:“通常我们会把context分割成多个小的文件,以便于管理。在测试时我们需要读取context文件,但是并不是每次都读取所有的文件。读取这些文件是很费时间的。所以我们想把它缓存起来,只要这个文件被读取过一次,我们就把它们缓存起来。所以我们通过扩展Junit的TestCase类来完成一个测试基类。我们需要实现缓存的逻辑,其它开发人员只需要实现读取配置文件的方法即可。它不用管是否具有缓存。”

代码:

public AbstractCacheContextTests extends TestCase{

private static Map contextMap=new HashMap();

protected ConfigurableApplicationContext applicationContext;

protected boolean hasCachedContext(Object contextKey){

return contextKeyToContextMap。containsKey(contextKey);

}

protected ConfigurableApplicationContext getContext(Object key){

String keyString=contextKeyString(key);

ConfigurableApplicationContext ctx=(ConfigurableApplicationContext)contextKeyToContextMap。get(keyString);

if(ctx……null){

if(key instanceof String[]){

ctx=loadContextLocations((String[])key);

}

contextKeyToContextMap。put(keyString,ctx);

}

return ctx;

}

protected String contextKeyString(Object contextKey){

if(contextKey instanceof String[]){

return StringUtils。arrayToCommaDelimitedString((String[])contextKey);

}

else{

return contextKey。toString();

}

}

protected ConfigurableApplicationContext loadContextLocations(String[]locations){

return new ClassPathXmlApplicationContext(locations);

}

//覆写TestCase的setUp方法,在运行测试方法之前从缓存中读取context文件,如果缓存中不存在则初始化applicationContext,并放入缓存。

protected final void setUp()throws Exception{

String[]contextFiles=getConfigLocations();

applicationContext=getContext(contextFiles);

}

//读取context文件,由子类实现

protected abstract String[]getConfigLocations();

}

大B:“rendercode();这样子类只需要去实现getConfigLocations方法,提供需要读取的配置文件字符数组就可以了。至于怎么去读取context文件内容,怎么实现缓存,则无需关心。AbstractCacheContextTests保证在运行所有测试之前去执行读取context文件的动作。注意这里setUp方法被声明为protected,是因为setUp方法是TestCase类的方法。在这里setUp方法被定义为final了,是确保子类不能去覆写这个方法,从而保证了父类控制的逻辑。”

小A:“如果使用过Junit会发生什么问题?”

大B:“TestCase的setUp方法,就是在这个测试类的测试方法运行之前作一些初始化动作。如创建一些所有测试方法都要用到的公共对象等。在这里把setUp方法声明为final之后,子类再也无法去扩展它了,子类同时还需要一些额外的初始化动作就无法实现了。可能你会说:‘把setUp方法的final修饰符去掉就可以了啊’。这样是可以的,但是去掉final修饰符后,子类是可以覆写setUp方法,而去执行一些额外的初始化。而可怕的是,父类从此失去了必须读取context文件及缓存context内容的逻辑。为了解决这个问题,可以实现一个空方法onSetUp。在setUp方法中调用onSetUp方法。这样子类就可以通过覆写onSetUp方法来进行额外的初始化。”

//覆写TestCase的setUp方法,在运行测试方法之前从缓存中读取context文件,如果缓存中不存在则初始化applicationContext,并放入缓存。

代码:

protected

n class=“keyword”>final void setUp()throws Exception{

String[]contextFiles=getConfigLocations();

applicationContext=getContext(contextFiles);

onSetUp();

}

protected void onSetUp(){

}

//读取context文件,由子类实现

protected abstract String[]getConfigLocations();

}

rendercode();

小A:“为什么不把onSetUp声明为abstract呢?”

大B:“这是因为子类不一定总是需要覆写onSetUp方法。可以说onSetUp方法是为了对setUp方法的扩展。像onSetUp这样的空方法就称之为勾子方法(HookMethod)。”