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

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. 78.3D立体轮播图(完整兼容手机端和pc端)

    效果呈现来源于http://www.jq22.com/demo/jR3DCarousel-master20160315/ 在此基础上改成需要的3个分类的3D图 由于原有的不支持粘贴复制显示3个分类 我 ...

  2. 在dockers中调试dump的dotnet程序

    其他调试参考文章 centos7使用lldb调试netcore应用转储dump文件 centos7 lldb 调试netcore应用的内存泄漏和死循环示例(dump文件调试) 生成dump文件 如何在 ...

  3. Docker在Centos 7上的部署

      Docker在Centos 7上的部署 方法1---开启centos 7上自带的 extras YUM源,然后 yum install docker来安装 安装前必须保证 Linux Kernel ...

  4. Nginx 高级配置-压缩功能

    Nginx 高级配置-压缩功能 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Nginx压缩相关参数概述 1>.gzip on | off; Nginx支持对指定类型的文 ...

  5. Leetcode——2. 两数相加

    难度: 中等 题目 You are given two non-empty linked lists representing two non-negative integers. The digit ...

  6. centos7下搭建JDK和Hadoop

    涉及基础操作命令 这里只是将涉及到的提了下一下具体的使用还需要读者自己查阅资料 tar 解压命令 su 进入root用户模式 rm -rf 删除 cd /文件名/.../ 进入某个文件夹下 注意要逐层 ...

  7. 201871010132——张潇潇《面向对象程序设计JAVA》第二周学习总结

    项目 内容 这个作业属于哪个课程 https://www.cnblogs.com/nwnu-daizh/ 这个作业的要求在哪里 https://www.cnblogs.com/nwnu-daizh/p ...

  8. 00-赵志勇机器学习-Logistics_Regression-data.txt(转载)

    4.45925637575900 8.22541838354701 0 0.0432761720122110 6.30740040001402 0 6.99716180262699 9.3133933 ...

  9. 第二阶段冲刺(个人)——five

    今天的计划:优化登录.注册信息的填写判断. 昨天做了什么?做背景. 困难:无

  10. First Chance Exception是什么?

    是否调试过应用程序并在输出窗口中看到有关“First Chance”异常的消息?有没有想过: 什么是First Chance Exception? 第一次机会异常是否意味着我的代码中存在问题? 在调试 ...