设计模式之责任链模式及其应用

news/2024/5/19 12:15:19 标签: 责任链模式

概述

责任链模式的定义:使多个对象都有机会处理请求,从而避免请求的发送者和接受者之间的耦合关系, 将这个对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理他为止。

应用

Servlet

Servlet里面的Filter便使用责任链模式,FilterChain

public final class ApplicationFilterChain implements FilterChain {
    private int pos = 0; // 当前执行filter的offset
    private int n; // 当前filter的数量
    private ApplicationFilterConfig[] filters; // filter配置类,通过getFilter()获取Filter
    private Servlet servlet;
  
    @Override
    public void doFilter(ServletRequest request, ServletResponse response) {
        if (pos < n) {
            ApplicationFilterConfig filterConfig = filters[pos++];
            Filter filter = filterConfig.getFilter();
            filter.doFilter(request, response, this);
        } else {
            // filter都处理完毕后,执行servlet
            servlet.service(request, response);
        }
    }
}

Tomcat

责任链模式是指在一个请求处理的过程中有很多处理者依次对请求进行处理,每个处理者负责做自己相应的处理,处理完之后将处理后的请求返回,再让下一个处理者继续处理。

责任链模式通常包含2个角色:

  • Handler(抽象处理者):定义一个处理请求的接口,即Container
  • ConcreteHandler(具体处理者):处理请求的具体类,或者传给下家,即Standardxxx

责任链模式是Tomcat中Container设计的基础,整个容器的就是通过一个链连接在一起,这个链一直将请求正确的传递给最终处理请求的那个Servlet。Container使用Pipeline-Value管道来处理请求,从Engine到Host到Context一直到Wrapper都是通过一个链传递请求。

Pipeline-Value使用的责任链模式和普通的责任链模式有两点不同:

  1. 每个Pipeline都有特定的Value,而且是在管道的最后一个执行,这个Value叫做BaseValue,BaseValue是不可删除的;
  2. 在上层容器的管道的BaseValue中会调用下层容器的管道。

Container包含四个子容器,而这四个子容器对应的BaseValue分别在:StandardEngineValve、StandardHostValve、StandardContextValve、StandardWrapperValve。
Pipeline的处理流程图如下:
在这里插入图片描述

  1. Connector在接收到请求后会首先调用最顶层容器的Pipeline来处理,这里的最顶层容器的Pipeline就是EnginePipeline;
  2. 在Engine的管道中依次会执行EngineValue1、EngineValue2等等,最后会执行StandardEngineValue,在StandardEngineValue中会调用Host管道,然后再依次执行Host的HostValue1、HostValue2等,最后在执行StandardHostValue,然后再依次调用Context的管道和Wrapper的管道,最后执行到StandardWrapperValue。
  3. 当执行到StandardWrapperValue的时候,会在StandardWrapperValue中创建FilterChain,并调用其doFilter方法来处理请求,这个FilterChain包含着我们配置的与请求相匹配的Filter和Servlet,其doFilter方法会依次调用所有的Filter的doFilter方法和Servlet的service方法,这样请求就得到处理!
  4. 当所有的Pipeline-Value都执行完之后,并且处理完具体的请求,这个时候就可以将返回的结果交给Connector,Connector通过Socket的方式将结果返回给客户端。

类图
在这里插入图片描述
其中有两个顶级接口Contained和Container,Contained提供操作Container的方法:

public interface Contained {
    Container getContainer();
    
    void setContainer(Container var1);
}

Spring

可同时配置多映射处理器,通过配置它们实现的Ordered 接口的 order 属性,DispatcherServlet 按顺序采用哪个映射处理器责任链模式

Dubbo

Dubbo在创建Filter时,通过把Filter封装成Invoker的匿名类,通过链表这样的数据结构来完成责任链,核心代码如下:

private static <T> Invoker<T> buildInvokerChain(final Invoker<T> invoker, String key, String group) {
    Invoker<T> last = invoker;
    //只获取满足条件的Filter
    List<Filter> filters = ExtensionLoader.getExtensionLoader(Filter.class).getActivateExtension(invoker.getUrl(), key, group);
    if (filters.size() > 0) {
        for (int i = filters.size() - 1; i >= 0; i --) {
            final Filter filter = filters.get(i);
            final Invoker<T> next = last;
            last = new Invoker<T>() {
                public Result invoke(Invocation invocation) throws RpcException {
                    return filter.invoke(next, invocation);
                }
            };
        }
    }
    return last;
}

Mybatis

Mybatis可以配置各种Plugin,Plugin和Filter类似,在执行SQL语句时做一些操作。Mybatis的责任链则是通过动态代理的方式,使用Plugin代理实际的Executor类。(这里实际还使用了组合模式,因为Plugin可以嵌套代理),核心代码如下:

public class Plugin implements InvocationHandler{
    private Object target;
    private Interceptor interceptor;
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {      
        if (满足代理条件) {
            return interceptor.intercept(new Invocation(target, method, args));
        }
        return method.invoke(target, args);     
    }
   
    //对传入的对象进行代理,可能是实际的Executor类,也可能是Plugin代理类
    public static Object wrap(Object target, Interceptor interceptor) {
        Class<?> type = target.getClass();
        Class<?>[] interfaces = getAllInterfaces(type, signatureMap);
        if (interfaces.length > 0) {
            return Proxy.newProxyInstance(type.getClassLoader(), interfaces,
                    new Plugin(target, interceptor, signatureMap));
        }
        return target;
    }
} 

netty

参考

责任链模式实现的三种方式


http://www.niftyadmin.cn/n/586406.html

相关文章

PHP 性能优化

1 不要随便就复制变量 有时候为了使 PHP 代码更加整洁&#xff0c;一些 PHP 新手&#xff08;包括我&#xff09;会把预定义好的变量复制到一个名字更简短的变量中&#xff0c;其实这样做的结果是增加了一倍的内存消耗&#xff0c;只会使程序更加慢。试想一下&#xff0c;在下面…

java ajax瀑布流,Ajax 实现滚动加载瀑布流 详解

/*** 需求&#xff1a;* 1、页面加载完成后 请求服务器 获取第一屏的数据* 2、点击加载更多 请求服务器 获取下一屏的数据* 3、当页面滚动到一定的高度时 请求服务器 获取下一屏的数据 渲染在页面中*/一、html代码加载更多二、js代码/*** 需求&#xff1a;* 1、页面加载完成后 …

[Visual Studio 2010] 敏捷利剑:详解Scrum

转载自http://news.csdn.net/a/20100308/217357.html 随着微软Visual Studio 2010 Ultimate Beta2版本的发布&#xff0c;除了它提供协同一致的ALM(应用程序生命周期)管理工具外&#xff0c;MSF for Agile Software Development过程框架从4.2升级到5.0&#xff0c;并且是以Scr…

Redis系列之过期淘汰机制

概述 使用redis时&#xff0c;一般是作为缓存系统&#xff0c;而不是存储系统。缓存系统&#xff0c;即需要设置一个生存时间&#xff08;TTL&#xff0c;time to live&#xff09;&#xff1b;存储系统&#xff0c;即不设置生存时间&#xff0c;永不过期。除了生存时间&#…

[Visual Studio 2010] 开发与测试敏捷特性

转载自IT168 在测试方面&#xff0c;VSTS 2010强化了测试功能&#xff0c;同时简化了在整个应用程序生命周期中整合测试的工具。新功能包括快速进行有关测试的设计与开发、测试用例管理&#xff0c;与Team Foundation Server集成的测试计划&#xff0c;以及确保所有更新的程序…

java ftp4j 限速,Java常用FTP文件操作说明 Apache.FTPClient,ftp4j,jftp

最近因工作需要&#xff0c;数据库中的数据需要从FTP服务中抽取数据文件然后校检再抽取到数据中。因为第一步需要从FTP服务中抽取数据文件。第二步采用JDBC批量数据更新。1。采用Apache.FTPClient:Java代码1. /**2. * Apache.FTPClient FTP操作共公类3. *4. * author 张明学5. …

MSN 9.0 同时登陆多个账号

From:http://kevinzhou.org/2009/01/23/msn-90-%E5%90%8C%E6%97%B6%E7%99%BB%E9%99%86%E5%A4%9A%E4%B8%AA%E8%B4%A6%E5%8F%B7/ 最近升级了MSN9.0&#xff0c;但是以前用的MSN Shell和Messenger Plus等支持多个账号登陆的第三方软件都不支持9.0。网上试了很多方法都不行&#x…

spring线程池的使用

为了尽量减少耗时操作对Action执行的影响&#xff0c;使用TaskExecutor线程池来管理耗时任务&#xff0c;作为后台进程执行&#xff0c;从而解决了问题。 场景&#xff1a; 使用了Struts和Spring&#xff0c;但Struts的Action并未交给Spring容器管理&#xff0c;Spring容器仅仅…