查看原文
其他

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

atheva Java后端 2020-08-21

作 者:atheva

来 源:cnblogs.com/lizo/p/7503862.html

广而告之:由于此订阅号换了个皮肤,系统自动取消了读者的公众号置顶。导致用户接受文章不及时。您可以打开订阅号,选择置顶(标星)公众号,重磅干货,第一时间送达!

责任链模式的定义:使多个对象都有机会处理请求,从而避免请求的发送者和接受者之间的耦合关系, 将这个对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理他为止。这里就不再过多的介绍什么是责任链模式,主要来说说java中如何编写。主要从下面3个框架中的代码中介绍。

  • servlet中的filter

  • dubbo中的filter

  • mybatis中的plugin 这3个框架在实现责任链方式不尽相同。

Servlet中分别定义了一个 Filter和FilterChain的接口,核心代码如下:

public final class ApplicationFilterChain implements FilterChain {    private int pos = 0;    private int n;    private ApplicationFilterConfig[] filters;  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 {            servlet.service(request, response);        }    }}

代码还算简单,结构也比较清晰,定义一个Chain,里面包含了Filter列表和servlet,达到在调用真正servlet之前进行各种filter逻辑。

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

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 Result invoke(Invocation invocation) throws RpcException {                    return filter.invoke(next, invocation);                }                ...            };        }    }    return last;}

Dubbo的责任链就没有类似FilterChain这样的类吧Filter和调用Invoker结合起来,而是通过创建一个链表,调用的时候我们只知道第一个节点,每个节点包含了下一个调用的节点信息。 这里的虽然Invoker封装Filter没有显示的指定next,但是通过java匿名类和final的机制达到同样的效果。

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);      }
    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;    }}

 简单的示意图如下:

这里简单介绍了Servlet、Dubbo、Mybatis对责任链模式的不同实现手段,其中Servlet是相对比较清晰,又易于实现的方式,而Dubbo和Mybatis则适合在原有代码基础上,增加责任链模式代码改动量最小的。

1. 如何在面试中介绍自己的项目经验?

2. 6 款火爆的 GitHub 项目推荐

3. 动态数据源切换、多数据源,读写分离

4. 29 个问答带你梳理 Spring !

    您可能也对以下帖子感兴趣

    文章有问题?点此查看未经处理的缓存