分享

请问责任链真的是一种设计模式吗

 chenge 2007-07-27





         

坛子上讨论设计模式的也挺多的,但是关于这个责任链模式还没有人提及,我对责任链模式也有些疑惑,我的疑惑是,责任链真的是一种设计模式吗,我为什么有这 种疑惑呢,因为在我用到和学习的责任链一共有3种,但是n多的书上就是那么一种最基础的责任链模式。在我的心里,只要能完成责任传递的都可以称之为责任 链。当然也有可能我的这种理解是错误的,但如果我的理解是错误的,那么就是说每种模式都是有固定形态的,简单的模式是有固定形态,这我承认,但是如果说所 有的模式都有固定的形态我不太相信,下面我就用文章来说明我的观点。

下面我详细说说我眼里的3种责任链模式:
1, 基本版责任链
这里所谓的基本责任链模式是指书上网上到处都有的责任链模式,其实这种模式网上到处都有,我就炒炒冷菜。让我们来看看下面这个关于它的uml图

图1,见附件
这是最简单的责任链模式,但是事实上我很少看到它在真实的系统中被使用(也有可能经验不足)。大家看到,我在图中给接口和类取的名字是Filter,没错,这个模式待会将会和正真的Filter相比较。下面我们就看看在spring中如何实现这么一种责任链模式。
首先是Filter的接口如下:

代码
  1. /** 
  2.  * @author 张荣华(ahuaxuan) 
  3.  * {@link http://ahuaxuan.} 
  4.  * @version $Id$ 
  5.  */  
  6. public interface Filter {  
  7.       
  8.     void executeFilter();  
  9.       
  10. }  

接着是一个抽象类,用来hold住下一个Filter实例的引用

代码
  1. /** 
  2.  * @author 张荣华(ahuaxuan) 
  3.  * {@link http://ahuaxuan.} 
  4.  * @version $Id$ 
  5.  */  
  6. public abstract class BaseFilter implements Filter {  
  7.       
  8.     private Filter nextFilter;  
  9.   
  10.     public void doNextFilter(){  
  11.         if (nextFilter != null) {  
  12.             nextFilter.executeFilter();  
  13.         } else {  
  14.             // do something you need here!  
  15.             System.out.println("there is no filter in the chain!!!!!!!!");  
  16.         }  
  17.     }  
  18.   
  19.     /** 
  20.      * 该方法用来注入下一个filter 
  21.      * @param nextFilter 
  22.      */  
  23.     public void setNextFilter(Filter nextFilter) {  
  24.         this.nextFilter = nextFilter;  
  25.     }  
  26. }  

接着就是3个具体的filter了。同样,让我们来看一下这3个具体的filter的代码:
1权限检查filter

代码
  1. /** 
  2.  * @author 张荣华(ahuaxuan) 
  3.  * {@link http://ahuaxuan.} 
  4.  * @version $Id$ 
  5.  */  
  6. public class AuthFilter extends BaseFilter {  
  7.   
  8.     public void executeFilter(){  
  9.         System.out.println("1------------check the user in this filter!");  
  10.         doNextFilter();  
  11.         System.out.println("2------------check the user in this filter!");  
  12.     }  
  13. }  

2url重写filter
代码
  1. /** 
  2.  * @author 张荣华(ahuaxuan) 
  3.  * {@link http://ahuaxuan.} 
  4.  * @version $Id$ 
  5.  */  
  6. public class URLRewriteFilter extends BaseFilter {  
  7.   
  8.     public void executeFilter(){  
  9.         System.out.println("1------------do url rewrite in this filter");  
  10.         doNextFilter();  
  11.         System.out.println("2------------do url rewrite in this filter");  
  12.     }  
  13. }  

3其他的filter
代码
  1. /** 
  2.  * @author 张荣华(ahuaxuan) 
  3.  * {@link http://ahuaxuan.} 
  4.  * @version $Id$ 
  5.  */  
  6. public class OtherFilter extends BaseFilter {  
  7.   
  8.     public void executeFilter(){  
  9.         System.out.println("1---------------do other things in this filter");  
  10.         doNextFilter();  
  11.         System.out.println("2---------------do other things in this filter");  
  12.     }  
  13. }  

3个具体的filter有了之后就等组装“链”了。我们看看spring中的配置:
代码
  1. <bean id="authFilter" class="org.easy.sample.designpattern.reschain.one.AuthFilter">  
  2.         <property name="nextFilter" value="urlRewriteFilter"></property>  
  3.     </bean>  
  4.     <bean id="urlRewriteFilter" class="org.easy.sample.designpattern.reschain.one.URLRewriteFilter">  
  5.         <property name="nextFilter" value="otherFilter"></property>  
  6.     </bean>  
  7. <bean id="otherFilter" class="org.easy.sample.designpattern.reschain.one.OtherFilter"/>  

O了,就这样,当你从spring的的context中得到这个authFilter的时候,你其实得到了一条链,当你调用authFilter的 方法是,这个调用会沿着链向下传递。到目前为止,我们已经实现了一个很搓(差)的责任链了,现在如果你在使用struts1.x系列,那么你就可以把这个 代码搬到struts1.x里,给action实现它自己的不依赖于servlet容器的拦截器了(当然如果是你真的想的话)。

众所周知的责任链成型了,但是你能忍受这种所谓的责任链模式吗,至少我不能,即使我使用struts1.x,我也不会傻到把上面这个破代码搬到项目里去的。那么就再让我们来看看第二种责任链模式吧

2, 混血版责任链(基于观察者的责任链)
我之前写了一篇关于观察者的文章( http://www./topic/102068 ),在这篇文章中,我曾经说过观察者稍微改一下就可以变成责任链,也就是说我们可以通过观察者来实现同上面一样的功能。这里我只简单描述一下如何用观察者来实现责任链。当然首先我们必须要理解观察者模式,如果不清楚观察者模式的就看http://www./topic/102068 这篇文章吧。

如果说之前的那个责任链是通过nextFilter引用而形成的话,那么观察者模式中的观察者的集合也可以看作是一个链,只不过这个链上的每一个 update方法都是要执行的,我们可以让它们某些执行某些不执行吗,当然可以了,只需要在写update方法的实现判断一下是否需要执行就行了,判断的 条件就是update传进去的,比如说被观察者如下:

代码
  1. /** 
  2.  * @author 张荣华(ahuaxuan) 
  3.  * @version $Id$ 
  4.  */  
  5. public class Subject extends Observable{  
  6.       
  7.     /** 
  8.      * 业务方法,一旦执行某个操作,则通知观察者 
  9.      */  
  10.     public void doBusiness(){  
  11.         if (true) {  
  12.             super.setChanged();  
  13.         }  
  14.           
  15.           
  16.         notifyObservers("mail");  
  17.         notifyObservers("jms");  
  18.     }  
  19.   
  20.       
  21.     public static void main(String [] args) {  
  22.         //创建一个被观察者  
  23.         Subject subject = new Subject();  
  24.           
  25.         //创建两个观察者  
  26.         Observer mailObserver = new MailObserver();  
  27.         Observer jmsObserver = new JMSObserver();  
  28.           
  29.         //把两个观察者加到被观察者列表中  
  30.         subject.addObserver(mailObserver);  
  31.         subject.addObserver(jmsObserver);  
  32.           
  33.         //执行业务操作  
  34.         subject.doBusiness();  
  35.           
  36.           
  37.     }  
  38. }  

两个观察者如下:

代码
  1. /** 
  2.  * @author 张荣华(ahuaxuan) 
  3.  * @version $Id$ 
  4.  */  
  5. public class JMSObserver implements Observer{  
  6.   
  7.     public void update(Observable o, Object arg) {  
  8.         if ("jms".equals(arg)) {  
  9.             System.out.println("发送消息给jms服务器的观察者已经被执行");  
  10.         }  
  11.     }  
  12. }  


代码
  1. /** 
  2.  * @author 张荣华(ahuaxuan) 
  3.  * @version $Id$ 
  4.  */  
  5. public class MailObserver implements Observer{  
  6.   
  7.     /** 
  8.      * 这个类取名为MailObserver,顾名思义,她是一个用来发送邮件的观察者 
  9.      */  
  10.     public void update(Observable o, Object arg) {  
  11.         if ("mail".equals(arg)) {  
  12.             System.out.println("发送邮件的观察者已经被执行");  
  13.         }  
  14.     }  
  15.   
  16. }  

这样的话,每次通知观察者,在观察者列表中只有一个或多个观察者被执行了,如此一来,也实现了第一中观察者的功能。那么如果要为 struts1.x实现不依赖于servlet容器的拦截器的时候我会用它吗?还是不会,因为虽然我也认为它是责任链,但事实上它更具备观察者的特征,比 如说有一个列表,执行的时候需要遍历列表,并执行列表中对象的方法,这明显是观察者的特征。所以说它更像观察者,如果说观察者是父亲,责任链是母亲,那这 个第二种责任链其实更像父亲。

虽然我还是不会用它来实现上面的需求,但是有一点我倒觉得这种用法比较像spring提供的事件机制,只不过在事件机制中是根据参数是否是属于某 一类型(指instanceOf方法)来判断事件是否应该发生的,不管怎么说,都是用观察者模式来实现的(貌似我又在谈观察者的重要性了)。

其实上面的这个equals判断是可以用正则表达式来代替,也就是说每一个update方法执行之前可以先用正则表达式匹配一下参数是否符合要求,这样灵活性就更大了,在第3种责任链中,我将会用正则表达式来判断责任的传递。

接下来轮到第3种责任链了(口干舌燥,喝点水吧),
3, 潇洒版责任链
我们来看看所谓的潇洒版的责任链模式,如下图所示:
图2,见附件

很明显,这个图复杂多了,可以说这个责任链和基本版的责任链有本质的区别,他不是一个责任对象引用另外一个责任对象这种形式,通过这种形式形成了一条看不 见的链。但是这够oo吗,所谓责任链,其实暗示着我们这里有两个类,一个是责任,一个是链,如果像基本版的责任链就只有一个责任类,通过责任之间的依赖形 成了一个链。显然这种方式明显比不上潇洒版责任链,在这种责任链中,责任是一个对象,链也是一个对象。

我们来仔细看一下上面这个图,责任就是ConcreateHandler,链就是ConcreateHandlerChain,HandlerConfig是对责任的配置类,事实上链和责任之间的关系是靠HandlerConfig联系起来的。

首先我们来看一下链对象中的executeHandler()方法,这个方法是整个责任链的核心:

 

代码
  1. private HandlerConfig [] handlerConfigs;   
  2. private int firstElement = -1;  
  3. private String doRegex;  
  4. /** 
  5.      * 这个方法是核心方法,它根据正则表达式来决定哪些filter需要被执行,哪些filter不需要被执行 
  6.      */  
  7.     public void executeHandler() {  
  8.         if (handlerConfigs == null) {  
  9.             // 调用业务方法  
  10.             return;  
  11.         }  
  12.           
  13.         ++firstElement;  
  14.         if (firstElement < handlerConfigs.length) {  
  15.             if (handlerConfigs[firstElement].getPattern().matcher(doRegex).matches()) {  
  16.                 handlerConfigs[firstElement].getHandler().executeHandler(this);  
  17.             } else {  
  18.                 this.executeHandler();  
  19.             }  
  20.         }  
  21.     }  
值得注意的是这个方法中有一个递规操作用,它的作用是当前handler不被执行的话就去判断是否执行下一个handler,我对递规还是有点怕的,大家有其他方法吗
我们再来看看责任对象中的executeHandler方法,这个方法就是每个责任对象的行为:
代码
  1. /** 
  2.  * @author 张荣华(ahuaxuan) 
  3.  * {@link http://ahuaxuan. } 
  4.  * @version $Id$ 
  5.  */  
  6. public class AuthTestHandler implements Handler {  
  7.   
  8.     public void executeHandler(HandlerChain handlerChain){  
  9.         System.out.println("1----------check the user in this filter!");  
  10.         handlerChain.executeHandler();  
  11.         System.out.println("2----------check the user in this filter!");  
  12.     }  
  13. }  

就是通过责任对象和链对象之间的重复调用,我们就实现了责任的传递,那为什么还需要一个config呢,这是因为责任链上的责任我认为应该是不带状 态的,这个用户在写自己的责任对象的时候不需要考虑它的配置,因为通常它的配置是写在配置文件中的。我们来看一下config类:

代码
  1. /** 
  2.  * @author 张荣华(ahuaxuan) 
  3.  * {@link http://ahuaxuan. } 
  4.  * @version $Id$ 
  5.  */  
  6. public class HandlerConfig {  
  7.     private Handler handler;  
  8.     private String handlerName;  
  9.     private String regex;  
  10. private Pattern pattern;  
  11. //省略getter和setter方法  
  12. }  

熟悉tomcat的同学大概已经猜出来了,这种责任链事实上就是tomcat中的filter,几乎一样的,那么让我们再看看测试方法:

代码
  1. public static void main(String[] args) {  
  2.           
  3.         HandlerConfig [] configs = new HandlerConfig[]{  
  4.             new HandlerConfig(), new HandlerConfig(), new HandlerConfig()  
  5.         };  
  6.         configs[0].setHandler(new AuthTestHandler());  
  7.         configs[0].setHandlerName("authFilter");  
  8.         configs[0].setRegex("\\d*");  
  9.           
  10.           
  11.         configs[1].setHandlerName("charsetFilter");  
  12.         configs[1].setHandler(new CharsetTestHandler());  
  13.         configs[1].setRegex("^[A-Za-z]+$");  
  14.           
  15.         configs[2].setHandlerName("otherFilter");  
  16.         configs[2].setHandler(new OtherTestHandler());  
  17.         configs[2].setRegex("\\d*");  
  18.           
  19.         System.out.println("chain1");  
  20.           
  21.         EasyHandlerChain chain = new EasyHandlerChain("21");  
  22.         chain.setHandlerConfigs(configs);  
  23.         chain.executeHandler();  
  24.           
  25.         System.out.println("");  
  26.         System.out.println("chain2");  
  27.       
  28.           
  29.         EasyHandlerChain chain1 = new EasyHandlerChain("asdf");  
  30.         chain1.setHandlerConfigs(configs);  
  31.         chain1.executeHandler();  
  32.     }  

上面一些config对象的创建可以放到框架中,每个线程都创建一个EasyHandlerChain就行了,当然我们也可以把config对象 创建所需要的配置放到annotation中去,但在这个例子中没有必要这样做。在测试方法中,每个config都设置了匹配表达式,每个chain对象 都设置了匹配的值,简直和tomcat的filter配置一摸一样是吗。

附件是上面3种责任链的源代码, 尤其是第3种,只看上面贴出来的代码不太容易理解,看看源代码,运行一下就明白了。

总结一下吧:我们可以看到,在责任链的实现中我们使用到3种模式,这也是我的疑惑,责任链是一种设计模式吗,我认为它不应该属于某一种模式,责任链是一种 思想,这种思想凌驾于模式之上,我们可以用模式来实现这种思想。这就是我想说的,如果你觉的这种理解是不正确的,请务必说明理由。









    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多