设计模式系列 - Chain of Responsibility模式

简介:

1、引入: 从击鼓传花谈起

  击鼓传花是一种热闹而又紧张的饮酒游戏。在酒宴上宾客依次坐定位置,由一人击鼓,击鼓的地方与传花的地方是分开的,以示公正。开始击鼓时,花束就开始依次传递,鼓声一落,如果花束在某人手中,则该人就得饮酒。
  假比说,贾母、贾赦、贾政、贾宝玉和贾环是五个参加击鼓传花游戏的传花者,他们组成一个环链。击鼓者将花传给贾母,开始传花游戏。花由贾母传给贾赦,由贾赦传给贾政,由贾政传给贾宝玉,又由贾宝玉传给贾环,由贾环传回给贾母,如此往复(见下图)。当鼓声停止时,手中有花的人就得执行酒令。
    
  图1、击鼓传花。
  击鼓传花便是责任链模式的应用。在责任链模式里,很多的对象由每一个对象对其下家的引用而联接起来形成一条链。请求在这个链上传递,直到链上的某一个对象决定处理此请求。发出这个请求的客户端并不知道链上的哪一个对象最终处理这个请求,这使得系统可以在不影响客户端的情况下动态地重新组织链和分配责任。
  责任链可能是一条直线、一个环链甚至一个树结构的一部分。
2、责任链模式的结构

 GoF对职责链模式的定义为:为了避免请求的发送者和接收者之间的耦合关系,使多个接受对象都有机会处理请求。将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。通俗一点说就是,当客户提交一个请求时,从第一个对象开始,链中收到请求的对象要么亲自处理它,要么转发给链中的下一个候选者。提交请求的对象并不知道哪一个对象将会处理它。当然职责链中也可能没有可以处理该请求的对象,这种情况是允许发生的。

 责任链模式是一种对象的行为模式,它所涉及到的角色如下:
  第一、抽象处理者(Handler)角色、定义出一个处理请求的接口;如果需要,接口可以定义出一个方法,以返回对下家的引用。下图给出了一个示意性的类图:
    
  图2、抽象处理者角色。 
  在图中的积累关系给出了具体子类对下家的引用,抽象方法handleRequest()规范了子类处理请求的操作。
  第二、具体处理者(ConcreteHandler)角色、处理接到请求后,可以选择将请求处理掉,或者将请 求传给下家。下图给出了一个示意性的类图。
   
   图3、具体处理者角色。 
  上图中的示意性的具体处理者ConcreteHandler类只有handleRequest()一个方法。
  责任链模式的静态类结构可见下图:
   
  图4、责任链模式的类图定义。

3、代码实例

 

 
  1. public interface Handler { 
  2.     public void handleRequest(String request); 

 

 
  1. public class HighRiskHandler implements Handler{ 
  2.     private static String[] riskWords= {"李刚","蒜你狠","豆你玩"}; 
  3.      
  4.     private Handler nextHandler; 
  5.  
  6.     public HighRiskHandler(Handler handler){ 
  7.         this.nextHandler = handler; 
  8.     } 
  9.  
  10.     public void handleRequest(String request){ 
  11.         //process request 
  12.         for(String risk : riskWords){ 
  13.             request = request.replaceAll(risk, ""); 
  14.         } 
  15.         // next processor 
  16.         nextHandler.handleRequest(request); 
  17.          
  18.     } 

 

 
  1. import org.slf4j.Logger; 
  2. import org.slf4j.LoggerFactory; 
  3.  
  4. public class LoggingFilter implements Handler { 
  5.     private static Logger logger = LoggerFactory.getLogger(LoggingFilter.class); 
  6.  
  7.     private Handler nextHandler; 
  8.  
  9.     public LoggingFilter(Handler handler) { 
  10.         this.nextHandler = handler; 
  11.     } 
  12.  
  13.     @Override 
  14.     public void handleRequest(String request) { 
  15.         logger.info("before " + request); 
  16.         this.nextHandler.handleRequest(request); 
  17.         logger.info("after " + request); 
  18.     } 
  19.  

 

 

【注意】

1、与观察者模式的区别

观察者模式重点在于变化是通知各个观察者,各观察者观察到的信息是相同的;而责任链模式在于使用链条的方式处理发送的内容,链条的前一环节处理之后的内容传递到下一环节,前面处理的结果可能会对后面的处理有影响

2、与装饰者模式的区别

装饰者模式重点在于加上各种修饰;责任链模式重在各个链条的处理,在调用的前后加上各种处理,中间可能存在堆栈调用的问题, 存在A前->B前->B后->A后的堆栈调用的情况

 

 本文转自 tianya23 51CTO博客,原文链接:http://blog.51cto.com/tianya23/536838,如需转载请自行联系原作者

相关文章
|
23天前
|
设计模式 SQL 算法
设计模式了解哪些,模版模式
设计模式了解哪些,模版模式
21 0
|
2月前
|
设计模式 Java uml
C++设计模式之 依赖注入模式探索
C++设计模式之 依赖注入模式探索
40 0
|
19天前
|
设计模式 Java 数据库
小谈设计模式(2)—简单工厂模式
小谈设计模式(2)—简单工厂模式
|
4天前
|
设计模式 消息中间件 Java
Java 设计模式:探索发布-订阅模式的原理与应用
【4月更文挑战第27天】发布-订阅模式是一种消息传递范式,被广泛用于构建松散耦合的系统。在 Java 中,这种模式允许多个对象监听和响应感兴趣的事件。
21 2
|
7天前
|
设计模式 存储 JavaScript
[设计模式Java实现附plantuml源码~创建型] 多态工厂的实现——工厂方法模式
[设计模式Java实现附plantuml源码~创建型] 多态工厂的实现——工厂方法模式
|
7天前
|
设计模式 Java Go
[设计模式Java实现附plantuml源码~创建型] 集中式工厂的实现~简单工厂模式
[设计模式Java实现附plantuml源码~创建型] 集中式工厂的实现~简单工厂模式
|
9天前
|
设计模式
设计模式(一)简单工厂模式
设计模式(一)简单工厂模式
14 0
|
19天前
|
设计模式 Java
小谈设计模式(9)—工厂方法模式
小谈设计模式(9)—工厂方法模式
|
28天前
|
设计模式 Java
23种设计模式,工厂方法模式的概念优缺点以及JAVA代码举例
【4月更文挑战第10天】工厂方法模式是设计模式中的一种创建型模式,它主要解决的问题是对象创建的问题。它定义了一个创建对象的接口,但让实现这个接口的类来决定实例化哪一个类。工厂方法让类的实例化推迟到子类中进行。
27 3
|
2月前
|
设计模式 编译器
解析器模式--设计模式
解析器模式--设计模式
19 0