漫画设计模式:什么是 “职责链模式” ?

2020-12-08 14:57:01 浏览数 (1)

————— 第二天 —————

————————————

在一家公司里,有三个程序员,他们的名字分别是小A、小B、小C:

有一天,公司新来的产品经理有一个新需求,但她不知道这个需求应该由谁来负责。于是,她首先找到了小A:

于是,产品经理又找到了小B:

于是,产品经理又找到了小C:

像这样,一个任务从小A转移到小B,从小B转移到小C,再由小C最终处理,形成了一个完整的任务处理链条:

在上面这个链条当中,包含着不同的任务处理者。面对一个新任务,每个任务处理者需要判断自己能否处理该任务,如果能处理,则处理并返回;如果不能处理,则转交给下一个任务处理者,直到某一个任务处理者最终完成处理。这就是职责链模式的核心思想。

代码语言:javascript复制
abstract public class Handler {

    protected Handler successor;

    public void setSuccessor(Handler successor) {
        this.successor = successor;
    }

    abstract String handleRequest(String msg);
}

从上面这个抽象类可以看出,每一个Handler对象都包含着一个successor成员,指向它的下一个任务处理者,就像链表节点的next指针一样。

代码语言:javascript复制
public class HandlerA extends Handler {
    @Override
    String handleRequest(String msg) {
        if(msg.contains("a")){
            msg = msg.replace('a', '*');
        } else if(this.successor != null){
            msg = this.successor.handleRequest(msg);
        }
        return msg;
    }
}

public class HandlerB extends Handler {
    @Override
    String handleRequest(String msg) {
        if(msg.contains("b")){
            msg = msg.replace('b', '*');
        } else if(this.successor != null){
            msg = this.successor.handleRequest(msg);
        }
        return msg;
    }
}

public class HandlerC extends Handler {
    @Override
    String handleRequest(String msg) {
        if(msg.contains("c")){
            msg = msg.replace('c', '*');
        } else if(this.successor != null){
            msg = this.successor.handleRequest(msg);
        }
        return msg;
    }
}

在这三个Handler实现类中,做了相似的判断:

如果传入的消息字符串包含某个字母,则把对应的字母替换成*。一旦某个Handler替换了自己所负责的字母,就直接结束整个链路;如果没有自己所负责的字母,则指定下一个Handler继续处理。

代码语言:javascript复制
public class Client {

    public static void main(String[] args) {
        Handler handlerA = new HandlerA();
        Handler handlerB = new HandlerB();
        Handler handlerC = new HandlerC();

        handlerA.setSuccessor(handlerB);
        handlerB.setSuccessor(handlerC);

        System.out.println(handlerA.handleRequest("apple"));
        System.out.println(handlerA.handleRequest("bicycle"));
        System.out.println(handlerA.handleRequest("color"));
    }
}

在客户端代码中,可以灵活设置整个链路和处理者的次序,然后直接调用第一个处理者的handleRequest方法,就相当于启动了整个链路。

做过Web开发的小伙伴都知道,当客户端对Web应用发出HTTP请求的时候,会首先经过Tomcat容器的一层层过滤器(Filter),过滤器会针对请求的访问权限、参数合法性等方面进行验证和过滤。

这一层一层过滤器的实现,就使用了职责链模式。

熟悉SpringMVC框架源码的小伙伴应该都知道,客户端的HTTP请求到了Web应用之后,会被SpringMVC框架的DispatcherServlet类进行分发,分发给Controller层的具体方法。

在进入Controller层的业务逻辑之前,以及执行完业务逻辑之后,该请求都会经过一系列的拦截器(Interceptor)。这一系列拦截器的处理流程,也同样是职责链模式的实现。

—————END—————

0 人点赞