从Servlet、Dubbo、Mybatis聊聊责任链究竟怎么用

导读:本篇文章讲解 从Servlet、Dubbo、Mybatis聊聊责任链究竟怎么用,希望对大家有帮助,欢迎收藏,转发!站点地址:www.bmabk.com

📖摘要


今天分享下 —— 从Servlet、Dubbo、Mybatis聊聊责任链究竟怎么用 的一些基本知识,欢迎关注!


🌂分享

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

这里多介绍什么是责任链模式有兴趣就百度一下,主要来说说java中如何编写。主要从下面3个框架中的代码中介绍。

  • servlet中的filter
  • dubbo中的filter
  • mybatis中的plugin

🌹servlet中的Filter

servlet 中分别定义了一个 FilterFilterChain的接口,核心代码如下:

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

}

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

在这里插入图片描述


💖Dubbo中的Filter

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

private static  Invoker buildInvokerChain(final Invoker invoker, String key, String group) {
    Invoker last = invoker;
    //只获取满足条件的Filter
    List 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 next = last;
            last = new Invoker() {
                ...
                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

Mybatis可以配置各种Plugin,无论是官方提供的还是自己定义的, PluginFilter 类似,就在执行 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;
    }
} 

简单的示意图如下:

在这里插入图片描述


🎉最后

  • 更多参考精彩博文请看这里:《陈永佳的博客》

  • 喜欢博主的小伙伴可以加个关注、点个赞哦,持续更新嘿嘿!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/97443.html

(0)
小半的头像小半

相关推荐

极客之音——专业性很强的中文编程技术网站,欢迎收藏到浏览器,订阅我们!