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

责任链模式

转载自这篇文章

责任链模式的定义:使多个对象都有机会处理请求,从而避免请求的发送者和接受者之间的耦合关系, 将这个对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理他为止。这里就不再过多的介绍什么是责任链模式,主要来说说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则适合在原有代码基础上,增加责任链模式代码改动量最小的。

  • 0
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java 责任链模式是一种设计模式,主要用于处理请求。在这种模式中,多个对象形成一个并依次处理请求,直到有一个对象可以处理它为止。Java 责任链模式虽然简单,但却非常强大,可以应用于许多场景。 Java 责任链模式三种实现方式: 1、基于 OOP 实现 这种实现方式是最基本的责任链模式,代码中将每个处理者定义为一个独立的类,并且将它们按照顺序连接在一起。 这种实现方式的优点是简单易懂,代码可读性高,容易扩展。缺点是类的数量可能会很多,需要提前确定好责任的完整结构,不太灵活。 2、基于注解实现 基于注解实现 Java 责任链模式,需要使用 Java 的反射机制。通过定义注解类型,在接口方法上添加注解,再通过反射来寻找哪些方法应该被执行,找到后按顺序执行。如果某一个处理器不需要被执行,就可以跳过它。 这种实现方式的优点是利用注解功能可以不用创建很多类,代码可读性较好。缺点是需要使用反射来执行方法,会影响性能,不太容易排错。 3、基于 Spring AOP 实现 基于 Spring AOP 实现 Java 责任链模式,需要使用 Spring 框架的 AOP 功能。利用 AOP,可以在不修改业务代码的情况下,动态地将处理器按照顺序连接在一起,然后依次执行。 这种实现方式的优点是简单、灵活,可以动态修改责任的结构。缺点是有一定的学习门槛,需要掌握 Spring AOP 的相关知识。 总结起来,根据具体情况选择不同的实现方式,都可以实现 Java 责任链模式。除了以上三种实现方式,还可以使用 lambda 表达式来实现责任

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值