• 行为型设计模式——责任链模式


    摘要

    责任链模式(Chain of responsibility pattern): 通过责任链模式, 你可以为某个请求创建一个对象链. 每个对象依序检查此请求并对其进行处理或者将它传给链中的下一个对象。

    一、责任链模式意图

    职责链模式(Chain Of Responsibility) 是一种行为设计模式, 允许你将请求沿着处理者链进行发送。 收到请求后, 每个处理者均可对请求进行处理, 或将其传递给链上的下个处理者。使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系。 将这个对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。

    二、责任链模式场景

    • 当程序需要使用不同方式处理不同种类请求, 而且请求类型和顺序预先未知时,可以使用责任链模式。
    • 当必须按顺序执行多个处理者时, 可以使用该模式。
    • 如果所需处理者及其顺序必须在运行时进行改变, 可以使用责任链模式。

    三、责任链模式类图

    1. 处理者 (Handler) 声明了所有具体处理者的通用接口。 该接口通常仅包含单个方法用于请求处理, 但有时其还会包含一个设置链上下个处理者的方法。

    2. 基础处理者 (Base Handler) 是一个可选的类, 你可以将所有处理者共用的样本代码放置在其中。通常情况下, 该类中定义了一个保存对于下个处理者引用的成员变量。 客户端可通过将处理者传递给上个处理者的构造函数或设定方法来创建链。 该类还可以实现默认的处理行为: 确定下个处理者存在后再将请求传递给它。

    3. 具体处理者 (Concrete Handlers) 包含处理请求的实际代码。 每个处理者接收到请求后, 都必须决定是否进行处理, 以及是否沿着链传递请求。处理者通常是独立且不可变的, 需要通过构造函数一次性地获得所有必要地数据。

    4. 客户端 (Client) 可根据程序逻辑一次性或者动态地生成链。 值得注意的是, 请求可发送给链上的任意一个处理者, 而非必须是第一个处理者。

    四、责任链模式实现

    1. public abstract class Handler {
    2. protected Handler successor;
    3. public Handler(Handler successor) {
    4. this.successor = successor;
    5. }
    6. protected abstract void handleRequest(Request request);
    7. }
    1. public class ConcreteHandler1 extends Handler {
    2. public ConcreteHandler1(Handler successor) {
    3. super(successor);
    4. }
    5. @Override
    6. protected void handleRequest(Request request) {
    7. if (request.getType() == RequestType.type1) {
    8. System.out.println(request.getName() + " is handle by ConcreteHandler1");
    9. return;
    10. }
    11. if (successor != null) {
    12. successor.handleRequest(request);
    13. }
    14. }
    15. }
    1. public class ConcreteHandler2 extends Handler{
    2. public ConcreteHandler2(Handler successor) {
    3. super(successor);
    4. }
    5. @Override
    6. protected void handleRequest(Request request) {
    7. if (request.getType() == RequestType.type2) {
    8. System.out.println(request.getName() + " is handle by ConcreteHandler2");
    9. return;
    10. }
    11. if (successor != null) {
    12. successor.handleRequest(request);
    13. }
    14. }
    15. }
    1. public class Request {
    2. private RequestType type;
    3. private String name;
    4. public Request(RequestType type, String name) {
    5. this.type = type;
    6. this.name = name;
    7. }
    8. public RequestType getType() {
    9. return type;
    10. }
    11. public String getName() {
    12. return name;
    13. }
    14. }
    1. public enum RequestType {
    2. type1, type2
    3. }
    1. public class Client {
    2. public static void main(String[] args) {
    3. Handler handler1 = new ConcreteHandler1(null);
    4. Handler handler2 = new ConcreteHandler2(handler1);
    5. Request request1 = new Request(RequestType.type1, "request1");
    6. handler2.handleRequest(request1);
    7. Request request2 = new Request(RequestType.type2, "request2");
    8. handler2.handleRequest(request2);
    9. }
    10. }

    五、责任链模式总结

    责任链模式、命令模式、中介者模式和观察者模式都是用于处理请求发送者和接收者之间的不同连接方式:

    • 责任链按照顺序将请求动态传递给一系列的潜在接收者, 直至其中一名接收者对请求进行处理。
    • 命令在发送者和请求者之间建立单向连接。
    • 中介者清除了发送者和请求者之间的直接连接, 强制它们通过一个中介对象进行间接沟通。
    • 观察者允许接收者动态地订阅或取消接收请求。

    责任链模式通常与组合模式结合使用。 在这种情况下, 叶子组件接收到请求后, 可以将请求沿包含全体父组件的链一直传递至对象树的底部。

    责任链模式使用命令模式实现,在这种情况下, 你可以对由请求代表的同一个上下文对象执行许多不同的操作。还有另外一种实现方式, 那就是请求自身就是一个命令对象。 在这种情况下, 你可以对由一系列不同上下文连接而成的链执行相同的操作。

    责任链与装饰器模式类结构非常相似。 两者都依赖递归组合将需要执行的操作传递给一系列对象。 但是, 两者有几点重要的不同之处。责任链管理者可以相互独立地执行一切操作, 还可以随时停止传递请求。 另一方面, 各种装饰可以在遵循基本接口的情况下扩展对象的行为。 此外, 装饰无法中断请求的传递。

    六、责任链模式开源示例

    责任链模式属于三种设计模式分类中行为型模式,责任链模式为请求对象创建一个接收者处理链,通常接收者中会存在下一个接收者的引用,如果当前接收者无法处理当前请求时,则将请求对象传递到下一个接收者进行处理,从而实现责任链,直到可以处理请求对象为止。责任链模式中请求发送者无需知道是哪个接收者处理的请求对象以及请求对象处理的具体过程,能够很好的将请求发送者和接收者进行解耦。责任链模式通常用于各种过滤器链、拦截处理链中,例如JavaWeb中的Filter过滤器等等。Spring框架中责任链模式的应用有很多,包括web请求设计,security安全权限设计,aop切面设计等等。

    SpringMVC中实际上是基于Servlet的框架,当客户端发送请求到web时,都会进入DispatcherServlet中,然后根据Servlet的生命周期去执行doService方法,在doService方法中有一个关键方法doDispatch,其中就运用到得了责任链模式来对进来的请求进行处理。

    1. protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
    2. HttpServletRequest processedRequest = request;
    3. //处理执行链
    4. HandlerExecutionChain mappedHandler = null;
    5. boolean multipartRequestParsed = false;
    6. WebAsyncManager asyncManager = WebAsyncUtils.getAsyncManager(request);
    7. try {
    8. ModelAndView mv = null;
    9. Exception dispatchException = null;
    10. try {
    11. processedRequest = checkMultipart(request);
    12. multipartRequestParsed = (processedRequest != request);
    13. //获取处理对象
    14. mappedHandler = getHandler(processedRequest);
    15. if (mappedHandler == null) {
    16. noHandlerFound(processedRequest, response);
    17. return;
    18. }
    19. HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());
    20. String method = request.getMethod();
    21. boolean isGet = HttpMethod.GET.matches(method);
    22. if (isGet || HttpMethod.HEAD.matches(method)) {
    23. long lastModified = ha.getLastModified(request, mappedHandler.getHandler());
    24. if (new ServletWebRequest(request, response).checkNotModified(lastModified) && isGet) {
    25. return;
    26. }
    27. }
    28. //责任链模式实现一:执行调用链的前置处理
    29. if (!mappedHandler.applyPreHandle(processedRequest, response)) {
    30. return;
    31. }
    32. //处理请求
    33. mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
    34. if (asyncManager.isConcurrentHandlingStarted()) {
    35. return;
    36. }
    37. applyDefaultViewName(processedRequest, mv);
    38. //责任链模式实现二:执行调用链的前置处理
    39. mappedHandler.applyPostHandle(processedRequest, response, mv);
    40. }
    41. catch (Exception ex) {
    42. dispatchException = ex;
    43. }
    44. catch (Throwable err) {
    45. dispatchException = new NestedServletException("Handler dispatch failed", err);
    46. }
    47. processDispatchResult(processedRequest, response, mappedHandler, mv, dispatchException);
    48. }
    49. catch (Exception ex) {
    50. triggerAfterCompletion(processedRequest, response, mappedHandler, ex);
    51. }
    52. catch (Throwable err) {
    53. triggerAfterCompletion(processedRequest, response, mappedHandler,
    54. new NestedServletException("Handler processing failed", err));
    55. }
    56. finally {
    57. if (asyncManager.isConcurrentHandlingStarted()) {
    58. if (mappedHandler != null) {
    59. mappedHandler.applyAfterConcurrentHandlingStarted(processedRequest, response);
    60. }
    61. }
    62. else {
    63. if (multipartRequestParsed) {
    64. cleanupMultipart(processedRequest);
    65. }
    66. }
    67. }
    68. }

    HandlerExecutionChain的责任链模式,将需要执行的HandlerInterceptor拦截器,添加在HandlerExecutionChain责任链的interceptorList集合中,然后依次执行HandlerInterceptor相应的处理方法。以preHandle方法为例,在处理方法中,根据返回的处理boolean判断是否要继续进行下面的HandlerInterceptor对象的处理,最终完成整个调用链的拦截处理。HandlerExecutionChain的实现实际上是责任链模式的一种变性用法,它减少了不同处理对象的引用依赖,采用集合的方式来存放处理链对象,降低了耦合度。

    1. //HandlerExecutionChain的applyPreHandle方法
    2. //责任链模式,将处理对象给接收者,并不关心内部如何处理与调用
    3. boolean applyPreHandle(HttpServletRequest request, HttpServletResponse response) throws Exception {
    4. for (int i = 0; i < this.interceptorList.size(); i++) {
    5. HandlerInterceptor interceptor = this.interceptorList.get(i);
    6. //调用拦截器的preHandle方法,执行所有拦截器的preHandle处理方法
    7. if (!interceptor.preHandle(request, response, this.handler)) {
    8. triggerAfterCompletion(request, response, null);
    9. return false;
    10. }
    11. this.interceptorIndex = i;
    12. }
    13. return true;
    14. }

    使用示例: 责任链模式在 Java 程序中并不常见, 因为它仅在代码与对象链打交道时才能发挥作用。该模式最流行的使用案例之一是在 GUI 类中将事件向上传递给父组件。 另一个值得注意的使用案例是依次访问过滤器。下面是该模式在核心 Java 程序库中的一些示例:

    识别方法: 该模式可通过一组对象的行为方法间接调用其他对象的相同方法来识别, 而且所有对象都会遵循相同的接口。

    博文参考

    行为型 - 责任链(Chain Of Responsibility) | Java 全栈知识体系

    设计模式之职责链模式 | DESIGN

  • 相关阅读:
    设计模式(2)-创建型模式
    【IEEE2017】RL:机器人库:一种面向对象的机器人应用程序的方法
    多态day02
    HTML网页制作代码——简约的旅游图文相册博客HTML模板(12页)HTML+CSS+JavaScript 静态HTML旅行主题网页作业
    页面中使用iframe
    Nginx之负载均衡upstream模块简介和使用
    react路由v6版本NavLink的两个小坑及解决
    如何在 Ubuntu上使用 Nginx 设置密码验证
    数仓学习笔记(1)——用户行为数据采集
    iNFTnews | 国内NFT发展仅限于数字藏品吗?
  • 原文地址:https://blog.csdn.net/weixin_41605937/article/details/133376446