在责任链模式里,很多对象由每一个对象对其下家的引用而连接起来形成一条链。请求在这个链上传递,直到链上的某一个对象决定处理此请求。发出请求的客户端并不知道链上的哪一个对象,这使得系统可以在不影响客户端的情况下动态地重新组织链和分配责任。

1.  从击鼓传花谈起

  击鼓传花是一种热闹而紧张的游戏。在酒宴上宾客依次坐定位置,由一人击鼓,击鼓的地方与传花的地方是分开的,以示公正。开始击鼓时,花束就开始依次传递,鼓声一落,如果花束在某人手中,则该人就得饮酒。

  比如说,贾母、贾赦、贾政、贾宝玉和贾环是五个参加击鼓传花游戏的传花者,他们组成一个环链。击鼓者将花传给贾母,开始传花游戏。花由贾母传给贾赦,由贾赦传给贾政,由贾政传给贾宝玉,又贾宝玉传给贾环,由贾环传回给贾母,如此往复,如下图所示。当鼓声停止时,手中有花的人就得执行酒令。

  击鼓传花便是责任链模式的应用。责任链可能是一条直线、一个环链或者一个树结构的一部分。

2.  责任链模式的结构

涉及的角色如下:

抽象处理者角色:定义出一个处理请求的接口。如果需要,接口可以定义一个方法,以设定和返回下一个处理者的引用。

具体处理这角色:这一角色收到请求后,可以选择处理掉请求或者将请求传递给下家。

代码如下:

抽象处理者:

public abstract class AbstractHandler {

    protected AbstractHandler nextHandler;

    public abstract void handleRequest();

    public AbstractHandler getNextHandler() {
return nextHandler;
} public void setNextHandler(AbstractHandler nextHandler) {
this.nextHandler = nextHandler;
} }

具体处理者:如果有下家就交给下家,否则自己处理。

public class ConcreteHandler extends AbstractHandler {

    @Override
public void handleRequest() {
if (getNextHandler() != null) {
System.out.println("交给下一个处理器");
getNextHandler().handleRequest();
} else {
System.out.println("自己处理");
}
} }

客户端代码:

public class Client {
public static void main(String[] args) {
AbstractHandler handler1 = new ConcreteHandler();
AbstractHandler handler2 = new ConcreteHandler(); handler1.setNextHandler(handler2);
handler1.handleRequest();
}
}

  此例子的逻辑非常简单,有下家就传给下家,没有就自己处理请求。

3.  纯的与不纯的责任链模式

  一个纯的责任链模式要求一个具体的处理者对象只能在两个行为中选择一个:一是承担责任,二是把责任推给下家。不允许出现某一个具体处理者对象在承担了一部分责任后又 把责任向下传的情况。

  在一个纯的责任链模式里面,一个请求必须被某一个处理者对象所接收;在一个不纯的责任链模式里面,一个请求可以最终不被任何接收端对象所接收。

  纯的责任链模式的实际例子很难找到,一般看到的例子均是不纯的责任链模式的实现。有些人认为不纯的责任链根本不是责任链模式,这也许是有道理的。但是在实际的系统里,纯的责任链很难找到。如果坚持责任链不纯便不是责任链模式,那么责任链模式便不会有太大意义了。

4.  责任链模式的应用-日志记录器

     我们创建抽象类 AbstractLogger,带有详细的日志记录级别。然后我们创建三种类型的记录器,都扩展了 AbstractLogger。每个记录器消息的级别是否属于自己的级别,如果是则相应地打印出来,否则将不打印并把消息传给下一个记录器。

结构如下:

(1)不纯的责任链模式

public abstract class AbstractLogger {

    public static int DEBUG = 7;
public static int INFO = 6;
public static int ERROR = 3; protected int level; // 责任链中的下一个元素
protected AbstractLogger nextLogger; public void setNextLogger(AbstractLogger nextLogger) {
this.nextLogger = nextLogger;
} public void logMessage(int level, String message) {
if (this.level <= level) {
write(message);
} if (nextLogger != null) {
nextLogger.logMessage(level, message);
}
} public abstract void write(String message); }
public class DebugLogger extends AbstractLogger {

    public DebugLogger() {
this.level = AbstractLogger.DEBUG; this.setNextLogger(new InfoLogger());
} @Override
public void write(String message) {
System.out.println("DebugLogger message - > " + message);
} }
public class InfoLogger extends AbstractLogger {

    public InfoLogger() {
this.level = AbstractLogger.INFO; this.setNextLogger(new ErrorLogger());
} @Override
public void write(String message) {
System.out.println("InfoLogger message - > " + message);
} }
public class ErrorLogger extends AbstractLogger {

    public ErrorLogger() {
this.level = AbstractLogger.ERROR; } @Override
public void write(String message) {
System.out.println("ErrorLogger message - > " + message);
} }

客户端代码:

public class Client {

    public static void main(String[] args) {
AbstractLogger logger = new DebugLogger(); AbstractLogger logger2 = new ErrorLogger(); String msg = "测试日志";
int level = AbstractLogger.INFO; logger.logMessage(level, msg);
System.out.println("=======");
logger2.logMessage(level, msg);
} }

结果:

InfoLogger message - > 测试日志
ErrorLogger message - > 测试日志
=======
ErrorLogger message - > 测试日志

(2)纯的责任链模式

  上面的日志记录是不纯的,由于InfoLogger已经记录过了,因此就不需要再传给ErrorLogger了,修改抽象处理器,如果处理完就停止:

public abstract class AbstractLogger {

    public static int DEBUG = 7;
public static int INFO = 6;
public static int ERROR = 3; protected int level; // 责任链中的下一个元素
protected AbstractLogger nextLogger; public void setNextLogger(AbstractLogger nextLogger) {
this.nextLogger = nextLogger;
} public void logMessage(int level, String message) {
if (this.level <= level) {
write(message);
return;
} if (nextLogger != null) {
nextLogger.logMessage(level, message);
}
} public abstract void write(String message); }

重新运行上面客户端代码,结果如下:

InfoLogger message - > 测试日志
=======
ErrorLogger message - > 测试日志

5.  责任链模式演变 (实际责任链模式用法)

  上文说了责任链模式一般是不纯的责任链模式,因此演变之后的责任链模式是维护一个处理器链条(内部维护一集合存储处理器),然后按照顺序去处理请求(靠集合中顺序维护链条关系),如下模拟servlet的Filter过滤器:

过滤器类:

public interface Filter {

    void doFilter(FilterChain filterChain);
}
public class LoginFilter implements Filter {

    @Override
public void doFilter(FilterChain filterChain) {
System.out.println("LoginFilter 开启运行"); filterChain.doFilter();
} }
public class EncodeFilter implements Filter {

    @Override
public void doFilter(FilterChain filterChain) {
System.out.println("EncodeFilter 开启运行"); filterChain.doFilter();
} }
public class OtherFilter implements Filter {

    @Override
public void doFilter(FilterChain filterChain) {
System.out.println("OtherFilter 开启运行"); filterChain.doFilter();
} }

处理器链:

public interface FilterChain {

    void doFilter();
}
import java.util.ArrayList;
import java.util.List; public class PlainFilterChain implements FilterChain { private List<Filter> list = new ArrayList<>(); private int index = 0; public PlainFilterChain add(Filter filter) {
this.list.add(filter);
return this;
} public void doFilter() {
if (index == list.size()) {
return; // 处理完成之后就返回
} Filter f = list.get(index);// 过滤器链按index的顺序拿到filter
index++;
f.doFilter(this);
}
}

客户端代码:

public class Client {

    public static void main(String[] args) {
Filter loginFilter = new LoginFilter();
Filter encodeFilter = new EncodeFilter();
Filter otherFilter = new OtherFilter(); PlainFilterChain filterChain = new PlainFilterChain();
filterChain.add(loginFilter);
filterChain.add(encodeFilter);
filterChain.add(otherFilter); filterChain.doFilter();
} }

结果:

LoginFilter 开启运行
EncodeFilter 开启运行
OtherFilter 开启运行

  补充:如果某个过滤器想停止后面链的执行,不调用filterChain.doFilter()即可,实际在Servlet的Filter运行机制也是如此。

6.  总结

意图:避免请求发送者与接收者耦合在一起,让多个对象都有可能接收请求,将这些对象连接成一条链,并且沿着这条链传递请求,直到有对象处理它为止。

主要解决:职责链上的处理者负责处理请求,客户只需要将请求发送到职责链上即可,无须关心请求的处理细节和请求的传递,所以职责链将请求的发送者和请求的处理者解耦了。

何时使用:在处理消息的时候以过滤很多道。

关键代码:Handler 里面聚合它自己,在 HandlerRequest 里判断是否合适,如果没达到条件则向下传递,向谁传递之前 set 进去。

应用实例: 1、红楼梦中的"击鼓传花"。 2、JS 中的事件冒泡。 3、JAVA WEB 中 Apache Tomcat 对 Encoding 的处理,Struts2 的拦截器,jsp servlet 的 Filter。

优点: 1、降低耦合度。它将请求的发送者和接收者解耦。 2、简化了对象。使得对象不需要知道链的结构。 3、增强给对象指派职责的灵活性。通过改变链内的成员或者调动它们的次序,允许动态地新增或者删除责任。 4、增加新的请求处理类很方便。

缺点: 1、不能保证请求一定被接收。 2、系统性能将受到一定影响,而且在进行代码调试时不太方便,可能会造成循环调用。 3、可能不容易观察运行时的特征,有碍于除错。

使用场景: 1、有多个对象可以处理同一个请求,具体哪个对象处理该请求由运行时刻自动确定。 2、在不明确指定接收者的情况下,向多个对象中的一个提交一个请求。 3、可动态指定一组对象处理请求。

责任链(ChainOfResponsibility)模式的更多相关文章

  1. Mybatis源码手记-从缓存体系看责任链派发模式与循环依赖企业级实践

    一.缓存总览 Mybatis在设计上处处都有用到的缓存,而且Mybatis的缓存体系设计上遵循单一职责.开闭原则.高度解耦.及其精巧,充分的将缓存分层,其独到之处可以套用到很多类似的业务上.这里将主要 ...

  2. 十四、ChainOfResponsibility 责任链模式

    设计: 代码清单: Trouble: public class Trouble { private int number; public Trouble(int number){ this.numbe ...

  3. chainOfResponsibility责任链模式

    责任链(Chain of Responsibility)模式 : 责任链模式是对象的行为模式.使多个对象都有机会处理请求,从而避免请求的发送者和接受者直接的耦合关系.将这些处理对象连成一条链,沿着这条 ...

  4. 学习笔记——责任链模式ChainOfResponsibility

    责任链模式,主要是通过自己记录一个后继者来判断当前的处理情况.Handler中,再增加一个方法用于设置后继对象,如SetHandler(Handler obj). 然后Handler类以其子类的处理方 ...

  5. 责任链模式(chainOfResponsibility)

    参考文章:http://wiki.jikexueyuan.com/project/design-pattern-behavior/chain-four.html 定义: 使多个对象都有机会处理请求,从 ...

  6. C#设计模式-责任链模式

    在现实生活中,有很多请求并不是一个人说了就算的,例如面试时的工资,低于1万的薪水可能技术经理就可以决定了,但是1万~1万5的薪水可能技术经理就没这个权利批准,可能就需要请求技术总监的批准,所以在面试的 ...

  7. C#设计模式(21)——责任链模式

    一.引言 在现实生活中,有很多请求并不是一个人说了就算的,例如面试时的工资,低于1万的薪水可能技术经理就可以决定了,但是1万~1万5的薪水可能技术经理就没这个权利批准,可能就需要请求技术总监的批准,所 ...

  8. 二十四种设计模式:责任链模式(Chain of Responsibility Pattern)

    责任链模式(Chain of Responsibility Pattern) 介绍为解除请求的发送者和接收者之间耦合,而使多个对象都有机会处理这个请求.将这些对象连成一条链,并沿着这条链传递该请求,直 ...

  9. 24种设计模式--责任链模式【Chain ofResponsibility Pattern】

    中国古代对妇女制定了“三从四德”的道德规范,“三从”是指“未嫁从父.既嫁从夫.夫死从子”,也就是说一个女性,在没有结婚的时候要听从于父亲,结了婚后听从于丈夫,丈夫死了还要听儿子的,举个例子来说,一个女 ...

随机推荐

  1. 结对项目(JAVA)

    项目成员: 邓镇港 3117004608 陈嘉欣 3117004604 一.Github项目地址: https://github.com/kestrelcjx/operation_expression ...

  2. OpenInstall实现APP无邀请码推广

    1.登录OpenInstall网站,这里会为你创建一个AppKey,而这个东西在web页面会用到. 2.在推广页面中加入推广下载. <script type="text/javascr ...

  3. CDA数据分析【数据收集】

    一.机器收集数据 机器收集数据会从不同角度对数据进行抓取和采集,与之前手动收集数据不同,机器收集数据不再是用小样本.特定样本来采集和分析整体数据,而是采用大样本或整体数据进行分析,这打破了原来的数据分 ...

  4. 【MySQL】备份和恢复

    语法 mysqldump -uslave -p -h127.0.0.1 --single-transaction --set-gtid-purged=OFF database1 table1 tabl ...

  5. 记录一个h5页面生成canvas画布做签名的js插件--signature_pad

    demo地址:https://jsfiddle.net/02dLn15g/5/ GitHub地址:https://github.com/szimek/signature_pad 配置项: dotSiz ...

  6. properties文件属性值通过xml文件为 java entity属性赋值

    一.通过xml配置文件进行赋值: 举个栗子,一目了然: 1.1 properties文件: 1.2 xml配置文件,将properties属性与java entity属性相关联:(这是一个新建的xml ...

  7. Codeforces Round #605 (Div. 3)

    地址:http://codeforces.com/contest/1272 A. Three Friends 仔细读题能够发现|a-b| + |a-c| + |b-c| = |R-L|*2 (其中L ...

  8. java、python及jmeter操作redis(接口自动化必备)

    redis是nosql数据库之一,其存储结构简单,提供高性能服务,所以在架构中是很常用的. 在做接口自动化测试过程中,有时也会涉及到redis,比如:发送短信验证码,我们做接口自动化测试,如何模拟发送 ...

  9. 23-C#笔记-正则表达式

    等用的时候,可以现查. 参考: http://www.runoob.com/csharp/csharp-regular-expressions.html

  10. 重新学习Spring注解——声明式事务

    36.声明式事务-环境搭建 37.声明式事务-测试成功 38.[源码]-声明式事务-源码分析 /** * 声明式事务: * * 环境搭建: * 1.导入相关依赖 * 数据源.数据库驱动.Spring- ...