设计模式之责任链模式

责任链模式是将请求的处理对象像一条链条组合起来,形成对象链。这样做的好处就是请求并不需要知道处理对象是哪一个,实现了请求和处理对象的解耦。

首先先看使用责任链经典的三个地方
1.servlet中的Filter

public class PassThroughFilterChain implements FilterChain {

    private Filter filter;

    private FilterChain nextFilterChain;

    private Servlet servlet;

    public PassThroughFilterChain(Filter filter, FilterChain nextFilterChain) {
        Assert.notNull(filter, "Filter must not be null");
        Assert.notNull(nextFilterChain, "'FilterChain must not be null");
        this.filter = filter;
        this.nextFilterChain = nextFilterChain;
    }
    public PassThroughFilterChain(Servlet servlet) {
        Assert.notNull(servlet, "Servlet must not be null");
        this.servlet = servlet;
    }

    public void doFilter(ServletRequest request, ServletResponse response) throws ServletException, IOException {
        if (this.filter != null) {
            this.filter.doFilter(request, response, this.nextFilterChain);
        }
        else {
            this.servlet.service(request, response);
        }
    }

}

FilterChain执行第一个filter时,执行filter.doFilter方法时会把下一个filter通过参数传递形式传入。如何维护filter链式关系,可以通过一个集合或者数组。比如说ApplicationFilterChain就维护了一个List<FilterConfig>数组,每次调用可以外部控制条用或者通过参数传递形式。


2.dubbo中的Filter(ProtocolFilterWrapper)

private static <T> Invoker<T> buildInvokerChain(final Invoker<T> invoker, String key, String group) {
        Invoker<T> last = invoker;
        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 Class<T> getInterface() {
                        return invoker.getInterface();
                    }

                    public URL getUrl() {
                        return invoker.getUrl();
                    }

                    public boolean isAvailable() {
                        return invoker.isAvailable();
                    }

                    public Result invoke(Invocation invocation) throws RpcException {
                        return filter.invoke(next, invocation);
                    }

                    public void destroy() {
                        invoker.destroy();
                    }

                    @Override
                    public String toString() {
                        return invoker.toString();
                    }
                };
            }
        }
        return last;
    }

原理就是新建一个invoker包装Filter,Filter.invoke方法会传入前一个invoker,循环调用就会形成一个调用链条。新建的invoker担任的角色就是包装filter,并且设置包装filter后置处理器。实际初始invoker就是执行终点。


3.mybatis中的interceptor

public class Plugin implements InvocationHandler {

  private Object target;
  private Interceptor interceptor;
  private Map<Class<?>, Set<Method>> signatureMap;

  private Plugin(Object target, Interceptor interceptor, Map<Class<?>, Set<Method>> signatureMap) {
    this.target = target;
    this.interceptor = interceptor;
    this.signatureMap = signatureMap;
  }

  public static Object wrap(Object target, Interceptor interceptor) {
    Map<Class<?>, Set<Method>> signatureMap = getSignatureMap(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;
  }

  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    try {
      Set<Method> methods = signatureMap.get(method.getDeclaringClass());
      if (methods != null && methods.contains(method)) {
        return interceptor.intercept(new Invocation(target, method, args));
      }
      return method.invoke(target, args);
    } catch (Exception e) {
      throw ExceptionUtil.unwrapThrowable(e);
    }
  }

  private static Map<Class<?>, Set<Method>> getSignatureMap(Interceptor interceptor) {
    Intercepts interceptsAnnotation = interceptor.getClass().getAnnotation(Intercepts.class);
    if (interceptsAnnotation == null) { // issue #251
      throw new PluginException("No @Intercepts annotation was found in interceptor " + interceptor.getClass().getName());      
    }
    Signature[] sigs = interceptsAnnotation.value();
    Map<Class<?>, Set<Method>> signatureMap = new HashMap<Class<?>, Set<Method>>();
    for (Signature sig : sigs) {
      Set<Method> methods = signatureMap.get(sig.type());
      if (methods == null) {
        methods = new HashSet<Method>();
        signatureMap.put(sig.type(), methods);
      }
      try {
        Method method = sig.type().getMethod(sig.method(), sig.args());
        methods.add(method);
      } catch (NoSuchMethodException e) {
        throw new PluginException("Could not find method on " + sig.type() + " named " + sig.method() + ". Cause: " + e, e);
      }
    }
    return signatureMap;
  }
}

mybatis通过动态代理来实现责任链模式。Mybatis支持对Executor、StatementHandler、PameterHandler和ResultSetHandler 接口进行拦截,也就是说会对这4种对象进行代理。mybatis会生成被拦截接口对象的动态代理类,动态代理类相当于是被拦截对象和拦截器的包装类,动态代理类有个target属性保存被拦截对象,执行时先执行拦截器方法,执行成功调用被拦截对象。动态代理类还可以被拦截再次生成新的动态代理类,每次生成新的都会保存上一个动态代理类,从而实现链式调用。



mybatis的分页就是通过拦截器的形式实现的。

@Intercepts({ @Signature(method = "prepare", type = StatementHandler.class, args = { Connection.class }) })
public class PageInterceptor implements Interceptor {

    private static final Logger logger = LoggerFactory.getLogger(PageInterceptor.class);

    @Override
    public Object plugin(Object target) {
        if (target instanceof StatementHandler) {
            return Plugin.wrap(target, this);
        } else {
            return target;
        }
    }

    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        RoutingStatementHandler handler = (RoutingStatementHandler) invocation.getTarget();
        StatementHandler delegate = (StatementHandler) ReflectUtils.getFieldValue(handler, "delegate");
        handlePage(invocation, delegate, delegate.getBoundSql());
        return invocation.proceed();
    }

    private void handlePage(Invocation invocation, StatementHandler delegate, BoundSql boundSql) throws SQLException {
        Page page = getPage(boundSql.getParameterObject());
        if (page == null) {
            return;
        }

        MappedStatement ms = (MappedStatement) ReflectUtils.getFieldValue(delegate, "mappedStatement");

        if (page.isCount()) {
            count(page, boundSql, ms, (Connection) invocation.getArgs()[0]);
        }
        handleOrder(boundSql, page.getOrderRules());

        String pageSql = new StringBuilder(boundSql.getSql()).append(" limit ").append(page.getStartIndex()).append(",").append(page.getPageSize()).toString();

        ReflectUtils.setFieldValue(boundSql, "sql", pageSql);
    }

    private void handleOrder(BoundSql boundSql, List<OrderRule> orderRules) throws SQLException {
        if (CollectionUtils.isEmpty(orderRules)) {
            return;
        }

        StringBuilder orderBuilder = new StringBuilder();
        for (OrderRule orderRule : orderRules) {
            if (orderBuilder.length() > 0) {
                orderBuilder.append(", ");
            }
            orderBuilder.append(orderRule.getProperty()).append(" ").append(orderRule.getType());
        }

        ReflectUtils.setFieldValue(boundSql, "sql", new StringBuilder(boundSql.getSql()).append(" order by ").append(orderBuilder).toString());
    }

    protected void count(Page page, BoundSql boundSql, MappedStatement mappedStatement, Connection connection) throws SQLException {
        String countSql = "select count(0) from (" + boundSql.getSql() + ") as temp_count_alias";

        List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();
        BoundSql countBoundSql = new BoundSql(mappedStatement.getConfiguration(), countSql, parameterMappings, boundSql.getParameterObject());
        ReflectUtils.setFieldValue(countBoundSql, "metaParameters", ReflectUtils.getFieldValue(boundSql, "metaParameters"));

        ParameterHandler parameterHandler = new DefaultParameterHandler(mappedStatement, boundSql.getParameterObject(), countBoundSql);
        PreparedStatement pstmt = null;
        ResultSet rs = null;
        try {
            pstmt = connection.prepareStatement(countSql);
            parameterHandler.setParameters(pstmt);
            rs = pstmt.executeQuery();
            if (rs.next()) {
                int totalCount = rs.getInt(1);
                page.setTotal(totalCount);
            }
        } finally {
            if (rs != null) {
                try {
                    rs.close();
                } catch (Exception e) {
                    logger.error("close resultSet error", e);
                }
            }
            if (pstmt != null) {
                try {
                    pstmt.close();
                } catch (Exception e) {
                    logger.error("close PreparedStatement error", e);
                }
            }
        }
    }

    @SuppressWarnings("rawtypes")
    private Page getPage(Object paramArgs) {
        if (paramArgs == null) {
            return null;
        }

        if (paramArgs instanceof Page) {
            return (Page) paramArgs;
        }

        if (paramArgs instanceof Map) {
            for (Object obj : ((Map) paramArgs).values()) {
                if (obj instanceof Page) {
                    return (Page) obj;
                }
            }
        }
        return null;
    }
}

拦截器比较重要的两个方法plugin、intercept。
plugin是对哪些情况进行拦截,拦截时会生成被拦截对象的动态代理类,intercept是实际拦截执行的操作,这个方法会最终调用invocation.proceed执行操作。
构成链式调用的关键是Plugin interceptor.intercept(new Invocation(target, method, args)); 动态代理类会将被代理对象传入拦截器方法,实际调用就是被拦截对象,当多次包装时,被代理对象仍然是个代理类,就构成了链式调用。

从上面三个例子可以看出来,实现责任链模式关键在于两个角色抽象处理类和具体处理类,类图如下


1.抽象处理类:主要包含指向下一个处理类的成员变量和一个处理请求的方法handleRequest。handleRequest思想就是如果当前满足条件就处理,否则让下一个处理类nextHandler去处理。
2.具体处理类:具体处理类主要是对具体的处理逻辑和处理要满足条件的具体实现。

下面以一个审批金额例子来简单说明这个模式的应用
员工审批金额不能超过500,领导审批金额不能超过1000,经理审批金额不能超过10000。
抽象员工处理类

public abstract class AbstrackClerk {
    private AbstrackClerk next;


    public AbstrackClerk getNext() {
        return next;
    }

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

    public abstract int getLimit();

    public abstract void approve();

    public final void approveRequest(BorrowRequest request) {
        if (getLimit() < request.getRequestMoney()) {
            if (next != null) {
                next.approveRequest(request);
            } else {
                throw new IllegalStateException("金额过大");
            }
        } else {
            this.approve();
        }
    }
}

职员处理类

public class Clerk extends AbstrackClerk {

    @Override
    public int getLimit() {
        return 500;
    }

    @Override
    public void approve() {
        System.out.println("clerk approve");
    }
}

领导具体处理类

public class Leader extends AbstrackClerk {
    @Override
    public int getLimit() {
        return 1000;
    }

    @Override
    public void approve() {
        System.out.println("leader approve");
    }
}

经理具体处理类

public class Leader extends AbstrackClerk {
    @Override
    public int getLimit() {
        return 1000;
    }

    @Override
    public void approve() {
        System.out.println("leader approve");
    }
}

Client类 设置具体处理类的后继关系

public class Client {
    public static void main(String[] args) {
        AbstrackClerk clerk = new Clerk();
        AbstrackClerk leader = new Leader();
        AbstrackClerk manager = new Manager();
        List<AbstrackClerk> clerks = Arrays.asList(clerk, leader, manager);
        for (int i = 0; i < clerks.size() - 1; i++) {
            clerks.get(i).setNext(clerks.get(i + 1));
        }
        clerk.approveRequest(new BorrowRequest(5000));
    }
}

目前自己还没有动手实现责任链场景,等以后遇到这边继续更新

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,084评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,623评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,450评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,322评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,370评论 6 390
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,274评论 1 300
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,126评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,980评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,414评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,599评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,773评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,470评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,080评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,713评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,852评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,865评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,689评论 2 354

推荐阅读更多精彩内容

  • 定义:使多个对象都有机会处理请求,从而避免了请求的发送者和接收者之间的耦合关系,将这些对象形成一链条,并沿着这条链...
    刘孙猫咪阅读 936评论 0 1
  • 设计模式之责任链模式 前言 之前觉得责任链模式是一种很高大上的模式,因为它不想策略模式那么随处可见,也不想模板模式...
    缄默的石头阅读 350评论 0 0
  • 一、 二、 三、内容 1. 责任链模式责任链模式:责任链模式就是很多对象由每个对象对其下家的引用而连接起来形成一条...
    吴世浩阅读 7,242评论 6 1
  • 距离余老离世已经半个多月了,他留下的精神财富依旧在传诵。如今我们在外头,你却在里头。一直在思考故乡对我们而言究竟是...
    赋风雪阅读 269评论 2 2
  • 我常常在想一个问题!如果当时妈妈做了我的后盾!我现在的生活是不是另一个场景!我到现在心里还是会有点怨!
    A曼达阅读 199评论 0 0