zoukankan      html  css  js  c++  java
  • java 责任链模式的三种实现

    责任链模式

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

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

    servlet中的Filter

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

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    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的匿名类,通过链表这样的数据结构来完成责任链,核心代码如下:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    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;
    }

      

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

    Mybatis中的Plugin

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

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    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;
        }

     简单的示意图如下:

    输入图片说明

    总结

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

    上面是转发别人的。

    下面是个人的模拟实现

    1.web filter中的过滤器很多人都了解,所以不做模拟

    2.dubbo中filter

    public void test() {
            List<Filter> lists = new ArrayList<>();
            lists.add(new SayWorldFilter());
            lists.add(new SayHelloFilter());
    
            Request request = new MyRequest();
    
            for (Filter filter : lists) {
                Request next = request;
                request = new Request() {
                    @Override
                    public String test() {
                        return filter.filter(next);
                    }
                };
            }
            request.test();
        }
    

    3.mybatis 中使用动态代理生产的包装类实现过滤

    Request request = (Request) (RequestWrap.wrap(new MyRequest(), new SayWorldFilter()));
            request = (Request) (RequestWrap.wrap(request, new SayHelloFilter()));
            request.test();
    

    具体代码见附件。

    https://files.cnblogs.com/files/z-test/filter.rar

     

  • 相关阅读:
    uva 11294 Wedding
    uvalive 4452 The Ministers’ Major Mess
    uvalive 3211 Now Or Later
    uvalive 3713 Astronauts
    uvalive 4288 Cat Vs. Dog
    uvalive 3276 The Great Wall Game
    uva 1411 Ants
    uva 11383 Golden Tiger Claw
    uva 11419 SAM I AM
    uvalive 3415 Guardian Of Decency
  • 原文地址:https://www.cnblogs.com/z-test/p/9319116.html
Copyright © 2011-2022 走看看