0%

【设计模式】Chain of Responsibility Pattern 责任链模式

责任链模式理解剖析以及应用

责任链模式

责任链模式(Chain of Responsibility Pattern)为请求创建了一个接收者对象的链。这种模式给予请求的类型,对请求的发送者和接收者进行解耦。这种类型的设计模式属于行为型模式。
在这种模式中,通常每个接收者都包含对另一个接收者的引用。如果一个对象不能处理该请求,那么它会把相同的请求传给下一个接收者,依此类推。

意图

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

解决问题

为了避免请求发送者与多个请求处理者耦合在一起,于是将所有请求的处理者通过前一对象记住其下一个对象的引用而连成一条链;当有请求发生时,可将请求沿着这条链传递,直到有对象处理它为止。
职责链上的处理者负责处理请求,客户只需要将请求发送到职责链上即可,无须关心请求的处理细节和请求的传递,所以职责链将请求的发送者和请求的处理者解耦了。

何时使用?

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

应用环境举例

  1. 红楼梦中的”击鼓传花”。
  2. JS 中的事件冒泡。
  3. JAVA WEB 中 Apache Tomcat 对 Encoding 的处理,Struts2 的拦截器,jsp servlet 的 Filter。

优点

  1. 降低了对象之间的耦合度。该模式使得一个对象无须知道到底是哪一个对象处理其请求以及链的结构,发送者和接收者也无须拥有对方的明确信息。

  2. 增强了系统的可扩展性。可以根据需要增加新的请求处理类,满足开闭原则。

  3. 增强了给对象指派职责的灵活性。当工作流程发生变化,可以动态地改变链内的成员或者调动它们的次序,也可动态地新增或者删除责任。

  4. 责任链简化了对象之间的连接。每个对象只需保持一个指向其后继者的引用,不需保持其他所有处理者的引用,这避免了使用众多的 if 或者 if···else 语句。

  5. 责任分担。每个类只需要处理自己该处理的工作,不该处理的传递给下一个对象完成,明确各类的责任范围,符合类的单一职责原则。

  6. 降低耦合度。它将请求的发送者和接收者解耦。

  7. 简化了对象。使得对象不需要知道链的结构。

  8. 增强给对象指派职责的灵活性。通过改变链内的成员或者调动它们的次序,允许动态地新增或者删除责任。

  9. 增加新的请求处理类很方便。

缺点

  1. 不能保证请求一定被接收。
  2. 系统性能将受到一定影响,而且在进行代码调试时不太方便,可能会造成循环调用。
  3. 可能不容易观察运行时的特征,有碍于除错。
  4. 不能保证每个请求一定被处理。由于一个请求没有明确的接收者,所以不能保证它一定会被处理,该请求可能一直传到链的末端都得不到处理。
  5. 对比较长的职责链,请求的处理可能涉及多个处理对象,系统性能将受到一定影响。
  6. 职责链建立的合理性要靠客户端来保证,增加了客户端的复杂性,可能会由于职责链的错误设置而导致系统出错,如可能会造成循环调用。

理解

类图

  • 抽象处理者(Handler)角色:定义一个处理请求的接口,包含抽象处理方法和一个后继连接。
  • 具体处理者(Concrete Handler)角色:实现抽象处理者的处理方法,判断能否处理本次请求,如果可以处理请求则处理,否则将该请求转给它的后继者。
  • 客户类(Client)角色:创建处理链,并向链头的具体处理者对象提交请求,它不关心处理细节和请求的传递过程。
  • 责任链模式的本质是解耦请求与处理,让请求在处理链中能进行传递与被处理;
  • 理解责任链模式应当理解其模式,而不是其具体实现。
  • 责任链模式的独到之处是将其节点处理者组合成了链式结构,并允许节点自身决定是否进行请求处理或转发,相当于让请求流动起来。

实现

  • 抽象处理者角色

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    abstract class Handler {
    private Handler next;

    public void setNext(Handler next) {
    this.next = next;
    }

    public Handler getNext() {
    return next;
    }

    //处理请求的方法
    public abstract void handleRequest(String request);
    }

  • 具体处理者角色1

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    class ConcreteHandler1 extends Handler {
    public void handleRequest(String request) {
    if (request.equals("one")) {
    System.out.println("具体处理者1负责处理该请求!");
    } else {
    if (getNext() != null) {
    getNext().handleRequest(request);
    } else {
    System.out.println("没有人处理该请求!");
    }
    }
    }
    }

  • 具体处理者角色2

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    class ConcreteHandler2 extends Handler {
    public void handleRequest(String request) {
    if (request.equals("two")) {
    System.out.println("具体处理者2负责处理该请求!");
    } else {
    if (getNext() != null) {
    getNext().handleRequest(request);
    } else {
    System.out.println("没有人处理该请求!");
    }
    }
    }
    }
  • 测试

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    public class ChainOfResponsibilityPattern {
    public static void main(String[] args) {
    //组装责任链
    Handler handler1 = new ConcreteHandler1();
    Handler handler2 = new ConcreteHandler2();
    handler1.setNext(handler2);
    //提交请求
    handler1.handleRequest("two");
    }
    }

    结果

    1
    具体处理者2负责处理该请求!

    实例

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

  • 创建抽象的记录器类。
    AbstractLogger.java

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    public abstract class AbstractLogger {
    public static int INFO = 1;
    public static int DEBUG = 2;
    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);
    }
    }

    abstract protected void write(String message);
    }
  • 创建扩展了该记录器类的实体类。
    ConsoleLogger.java

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    public class ConsoleLogger extends AbstractLogger {

    public ConsoleLogger(int level){
    this.level = level;
    }

    @Override
    protected void write(String message) {
    System.out.println("Standard Console::Logger: " + message);
    }
    }

    ErrorLogger.java

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    public class ErrorLogger extends AbstractLogger {

    public ErrorLogger(int level){
    this.level = level;
    }

    @Override
    protected void write(String message) {
    System.out.println("Error Console::Logger: " + message);
    }
    }

    FileLogger.java

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    public class FileLogger extends AbstractLogger {

    public FileLogger(int level){
    this.level = level;
    }

    @Override
    protected void write(String message) {
    System.out.println("File::Logger: " + message);
    }
    }
  • 测试
    ChainPatternDemo.java

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22

    public class ChainPatternDemo {

    private static AbstractLogger getChainOfLoggers() {

    AbstractLogger errorLogger = new ErrorLogger(AbstractLogger.ERROR);
    AbstractLogger fileLogger = new FileLogger(AbstractLogger.DEBUG);
    AbstractLogger consoleLogger = new ConsoleLogger(AbstractLogger.INFO);

    errorLogger.setNextLogger(fileLogger);
    fileLogger.setNextLogger(consoleLogger);

    return errorLogger;
    }

    public static void main(String[] args) {
    AbstractLogger loggerChain = getChainOfLoggers();
    loggerChain.logMessage(AbstractLogger.INFO, "aaaaaaaaaaaaaaaaaaaaaaaa");
    loggerChain.logMessage(AbstractLogger.DEBUG, "bbbbbbbbbbbbbbbbbbbbbbbbbbbbb");
    loggerChain.logMessage(AbstractLogger.ERROR, "ccccccccccccccccccccccccccc");
    }
    }

    此时对象为

    结果:
    1
    2
    3
    4
    5
    6
    Standard Console::Logger: aaaaaaaaaaaaaaaaaaaaaaaa
    File::Logger: bbbbbbbbbbbbbbbbbbbbbbbbbbbbb
    Standard Console::Logger: bbbbbbbbbbbbbbbbbbbbbbbbbbbbb
    Error Console::Logger: ccccccccccccccccccccccccccc
    File::Logger: ccccccccccccccccccccccccccc
    Standard Console::Logger: ccccccccccccccccccccccccccc

感谢查阅