Java设计模式之责任链模式详细讲解和案例示范

news/2024/9/19 0:42:41/ 标签: java, 设计模式, 责任链模式, 系统架构, 面试, uml

在本文中,我们将详细讲解Java设计模式中的责任链模式,探讨其基本概念、使用场景、常见问题和解决方式。同时,我们还会介绍责任链模式与策略模式的区别,并结合电商交易系统的示例进行说明。此外,我们还会探讨责任链模式在开源框架中的应用。

1. 责任链模式概述

责任链模式是一种行为设计模式,旨在将请求沿着链传递,直到被某个处理器处理。它使得多个对象都有机会处理请求,避免了请求发送者与接收者的耦合。

1.1 主要特点
  • 请求传递:请求沿着链条依次传递,直到找到能够处理该请求的处理器。
  • 灵活性:可以动态增加或删除处理器,调整处理链的顺序。
  • 解耦:请求发送者和处理者之间没有直接的依赖关系。
1.2 结构

责任链模式主要包括以下几个角色:

  • Handler:定义了处理请求的接口,并包含了对下一个处理器的引用。
  • ConcreteHandler:具体的处理器类,负责处理具体的请求。
  • Client:发出请求的客户端,依赖于Handler接口。
2. 使用场景
  • 请求需要多个对象进行处理:例如,在订单处理系统中,不同的订单状态可能由不同的处理器进行处理。
  • 动态的请求处理链:如审批流程,处理链可以根据业务需求进行动态配置。

3. 电商交易系统中的示例

以电商交易系统为例,我们将演示如何使用责任链模式来处理订单状态的变化。例如,订单可能会经历“已支付”、“已发货”、“已收货”等状态,每个状态的处理都可以看作是责任链中的一个环节。

3.1 错误示范

在没有使用责任链模式的情况下,订单状态的处理可能会写成一长串的if-else语句,这样的代码可读性差,维护困难。

public class OrderService {public void processOrder(Order order) {if (order.getStatus().equals("PAID")) {// 处理已支付的订单} else if (order.getStatus().equals("SHIPPED")) {// 处理已发货的订单} else if (order.getStatus().equals("DELIVERED")) {// 处理已收货的订单}// 其他状态处理}
}
3.2 正确示范

使用责任链模式可以将不同状态的处理分离到各自的处理器中,增强代码的灵活性和可维护性。

abstract class OrderHandler {protected OrderHandler nextHandler;public void setNextHandler(OrderHandler nextHandler) {this.nextHandler = nextHandler;}public abstract void handleOrder(Order order);
}class PaidOrderHandler extends OrderHandler {@Overridepublic void handleOrder(Order order) {if (order.getStatus().equals("PAID")) {// 处理已支付的订单} else if (nextHandler != null) {nextHandler.handleOrder(order);}}
}class ShippedOrderHandler extends OrderHandler {@Overridepublic void handleOrder(Order order) {if (order.getStatus().equals("SHIPPED")) {// 处理已发货的订单} else if (nextHandler != null) {nextHandler.handleOrder(order);}}
}class DeliveredOrderHandler extends OrderHandler {@Overridepublic void handleOrder(Order order) {if (order.getStatus().equals("DELIVERED")) {// 处理已收货的订单} else if (nextHandler != null) {nextHandler.handleOrder(order);}}
}

使用责任链模式后,我们可以灵活地添加、删除或调整处理器顺序,无需修改现有代码。

3.3 类图

在这里插入图片描述

4. 常见问题与解决方式

4.1 问题一:责任链过长导致性能问题

在某些场景下,责任链可能会变得过长,导致每次请求传递的效率低下。

解决方式

  • 优化链条顺序:将最常用的处理器放在链条前面。
  • 分拆责任链:将责任链拆分成多个子链,分别处理不同的请求。
4.2 问题二:处理器无法处理请求

当所有处理器都无法处理请求时,可能会导致请求被忽略。

解决方式

  • 添加默认处理器:在链的末尾添加一个默认的处理器,处理无法被处理的请求。
class DefaultOrderHandler extends OrderHandler {@Overridepublic void handleOrder(Order order) {// 处理无法处理的请求}
}

5. 责任链模式与策略模式的区别

责任链模式与策略模式都是行为型模式,但它们的使用场景和目的有所不同。

  • 责任链模式:关注请求的传递,通过一系列的处理器逐个处理请求。适用于多个对象依次处理请求的场景。
  • 策略模式:关注算法的替换,通过定义一组算法类来实现相同的功能。适用于需要动态切换算法的场景。

6. 责任链模式在开源框架中的应用

责任链模式在许多开源框架中得到了广泛应用,例如Spring Security中的过滤器链就是责任链模式的典型应用。

6.1 Spring Security过滤器链简介

Spring Security是一个为Java企业应用提供安全功能的框架。它的核心之一就是过滤器链,用来对请求进行认证和授权。每个请求在进入应用程序之前,都会经过一组过滤器。每个过滤器依次处理请求,完成特定的安全检查任务,如身份验证、权限检查、跨站点请求伪造(CSRF)防护等。

这些过滤器形成了一个“责任链”,请求在链上一个一个过滤器中传递,直到某个过滤器处理完毕或整个链处理结束。

Spring Security的过滤器链是基于Servlet API中的javax.servlet.Filter接口构建的。该接口的核心方法是doFilter(),它接受请求和响应作为参数,并负责将请求传递给链中的下一个过滤器。

6.2 责任链模式的应用结构

责任链模式中,通常会有以下几个角色:

  • 处理者(Handler): 定义一个处理请求的接口。每个具体的处理者实现该接口来处理部分请求。
  • 具体处理者(Concrete Handler): 处理链中实际的处理器,处理请求或将请求传递给下一个处理者。
  • 客户(Client): 发起请求的对象,不关心谁处理了请求,也不需要知道具体的处理者。

在Spring Security中,处理者相当于Filter,而具体处理者是各种具体的安全过滤器,如UsernamePasswordAuthenticationFilterBasicAuthenticationFilter等。客户是HTTP请求。

6.3 Spring Security责任链模式工作流程

当客户端请求到达Spring Security的过滤器链时,过滤器链按照顺序执行每一个过滤器,直到某个过滤器决定阻止请求继续处理或最后一个过滤器处理完请求。

流程大致如下:

  1. 客户端发起请求:请求首先被交给第一个过滤器处理。
  2. 过滤器链依次执行:每个过滤器执行其doFilter()方法,执行安全检查。
  3. 请求传递:如果当前过滤器无法处理请求,或者处理完毕后仍需进一步检查,则将请求传递给下一个过滤器。
  4. 终止或继续:若某个过滤器决定终止请求(例如认证失败),则直接返回响应,不再执行后续过滤器;否则,继续传递请求。
  5. 最后一个过滤器完成处理:所有过滤器执行完毕后,若没有过滤器拦截请求,则请求到达目标资源。
6.4 案例:自定义过滤器链

现在我们通过一个实际的案例,演示如何在Spring Security中使用自定义过滤器链。假设我们需要在现有的Spring Security过滤器链中增加一个自定义的认证过滤器,用来处理特殊的请求认证逻辑。

步骤1:创建自定义过滤器 我们首先创建一个继承自OncePerRequestFilter的过滤器,并在其中实现自定义的认证逻辑。

import javax.servlet.FilterChain;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.security.core.context.SecurityContextHolder;
import org.springframework.web.filter.OncePerRequestFilter;import java.io.IOException;public class CustomAuthenticationFilter extends OncePerRequestFilter {@Overrideprotected void doFilterInternal(HttpServletRequest request,HttpServletResponse response,FilterChain filterChain)throws ServletException, IOException {String token = request.getHeader("Authorization");// 简单的认证逻辑:如果存在token,就通过认证if (token != null && token.startsWith("Bearer ")) {// 模拟将用户信息存入SecurityContextSecurityContextHolder.getContext().setAuthentication(new CustomAuthenticationToken(token));}// 继续传递请求到下一个过滤器filterChain.doFilter(request, response);}
}

在这个过滤器中,我们从HTTP请求的头部获取Authorization信息,并检查它是否以Bearer 开头。如果符合条件,就将认证信息存储在SecurityContextHolder中。之后,调用filterChain.doFilter()将请求继续传递给下一个过滤器。

步骤2:配置自定义过滤器链

接下来,我们需要将自定义的过滤器加入到Spring Security的过滤器链中。通过扩展WebSecurityConfigurerAdapter并重写configure方法,我们可以将自定义过滤器插入到合适的位置。

import org.springframework.context.annotation.Configuration;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
import org.springframework.security.config.annotation.web.configuration.WebSecurityConfigurerAdapter;
import org.springframework.security.web.authentication.UsernamePasswordAuthenticationFilter;@Configuration
public class SecurityConfig extends WebSecurityConfigurerAdapter {@Overrideprotected void configure(HttpSecurity http) throws Exception {http.addFilterBefore(new CustomAuthenticationFilter(), UsernamePasswordAuthenticationFilter.class).authorizeRequests().anyRequest().authenticated();}
}

在此配置中,我们使用了addFilterBefore()方法,将自定义的CustomAuthenticationFilter添加到UsernamePasswordAuthenticationFilter之前。这样,自定义过滤器将会在用户名密码认证过滤器之前执行。

步骤3:测试过滤器链

至此,过滤器链已经配置完成。我们可以通过HTTP请求来测试我们的过滤器链是否正常工作。

curl -H "Authorization: Bearer abc123" http://localhost:8080/api/orders

当客户端发送带有Authorization头的请求时,自定义过滤器会处理认证逻辑,随后将请求继续传递给下一个过滤器。如果认证通过,Spring Security将允许访问资源,否则将会返回未认证错误。

通过这种设计,Spring Security可以灵活地配置过滤器的顺序和内容,增强系统的可扩展性。

7. 总结

责任链模式在处理复杂的请求传递场景中非常有效,尤其是在需要灵活调整处理顺序的系统中。通过本篇文章的介绍,你应该已经掌握了责任链模式的基本概念、使用场景、常见问题以及如何在实际项目中应用它


http://www.ppmy.cn/news/1526113.html

相关文章

Basler 相机与LabVIEW进行集成

Basler 提供的相机驱动和 SDK (Software Development Kit) 允许用户通过 LabVIEW 对相机进行控制和图像采集。以下是 Basler 相机与 LabVIEW 集成的几种方式: 1. Baslers Pylon SDK Basler 提供的 Pylon SDK 是一套用于控制 Basler 相机的开发工具包,支…

ubuntu 安装 nvidia-docker

安装 nvidia-docker 之前要先安装好显卡驱动和docker,不需要安装CUDA 1、安装docker 先卸载旧版本docker sudo apt-get remove docker docker-engine docker.io containerd runc再进行 sudo apt-get update sudo apt-get install ca-certificates curl gnupg lsb…

ModuleNotFoundError: No module named ‘datasets‘

报错信息: 解决:安装datasets 方法1: pip install datasets 方法2: python3可以使用以下命令: pip3 install datasets

15.7 创建prometheus的statsfulset配置

本节重点介绍 : prometheus statsfulset yaml配置 设置statsfulset副本反亲和性设置pod运行优先级设置volumeClaimTemplates设置配置文件热更新容器 configmap-reload设置prometheus主容器 statsfulset 设置元信息 apiVersion: apps/v1 kind: StatefulSet metadata:name: p…

Rust:Result 和 Error

在 Rust 编程语言中,错误处理是一个核心部分,用于确保程序的健売性和可靠性。Rust 通过 Result 枚举和 Error 特质(trait)来处理错误。 Result 枚举 Result 是一个泛型枚举,用于表示一个操作可能成功或失败。它有两个…

时序预测|基于小龙虾优化高斯过程GPR数据回归预测Matlab程序COA-GPR 多特征输入单输出 附赠基础GPR

时序预测|基于小龙虾优化高斯过程GPR数据回归预测Matlab程序COA-GPR 多特征输入单输出 附赠基础GPR 文章目录 一、基本原理二、实验结果三、核心代码四、代码获取五、总结 时序预测|基于小龙虾优化高斯过程GPR数据回归预测Matlab程序COA-GPR 多特征输入单输出 附赠基础GPR 一、…

实时(按帧)处理的低通滤波C语言实现

写在前面: 低通滤波采用一般的FIR滤波器,因为本次任务,允许的延迟较多,或者说前面损失的信号可以较多,因此,涉及一个很高阶的FIR滤波器,信号起始段的信号点可以不处理,以及&#xf…

Spring boot中常用注解解释

Data 是Lombok提供的注解,结合了以下几个常用注解的功能: Getter: 自动为所有字段生成getter方法。 Setter: 自动为所有字段生成setter方法。 ToString: 自动生成toString()方法。 EqualsAndHashCode: 自动生成equals()和hashCode()方法。 RequiredArgs…

HTML5中`<span>`标签深入解析

引言 在HTML5中&#xff0c;<span>标签是一个行内元素&#xff0c;用于对文档中的一小部分文本或内容进行分组&#xff0c;以便于应用CSS样式或JavaScript脚本。与块级元素&#xff08;如<div>&#xff09;不同&#xff0c;<span>不会打断文本的流动&#x…

设计模式之单例模式(通俗易懂--代码辅助理解【Java版】)

文章目录 设计模式概述1、单例模式概述2、懒汉式&#xff1a;3、饿汉式4、懒汉式&#xff1a;解决反射、序列化反序列化问题5、懒汉式DCL&#xff08;推荐&#xff09;6、应用场景7、单例线程池实现8、总结 设计模式概述 创建型模式&#xff1a;工厂方法、抽象方法、建造者、原…

conda、anaconda、pip、torch、pytorch、tensorflow到底是什么东西?(转载自本人的知乎回答)

转载自本人的知乎回答&#xff08;截止2024年9月&#xff0c;1700赞同&#xff0c;2400收藏&#xff09; https://www.zhihu.com/question/566592612/answer/3063465880 如果你是一个大四的CS准研究生回去补基础课&#xff0c;假如是科班CS甚至科班EE的话那你基础也太差了。你…

相亲交友程序系统开发产品分析

相亲交友系统是一种专门为单身人士设计的社交平台&#xff0c;旨在帮助他们找到合适的伴侣。这类系统通常包括了线上和线下的多种互动方式&#xff0c;能够让参与者在舒适的环境中相识、相知。编辑&#xff1a;qawsed2466。以下是相亲交友系统的一些关键特点和优势&#xff1a;…

AI算法部署方式对比分析:哪种方案性价比最高?

随着人工智能技术的飞速发展&#xff0c;AI算法在各个领域的应用日益广泛。AI算法的部署方式直接关系到系统的性能、实时性、成本及安全性等多个方面。本文将探讨AI算法分析的三种主要部署方式&#xff1a;本地计算、边缘计算和云计算&#xff0c;并详细分析它们的优劣性。 一、…

计算机毕业设计选题推荐-推拿知识互动平台-Java/Python项目实战

✨作者主页&#xff1a;IT毕设梦工厂✨ 个人简介&#xff1a;曾从事计算机专业培训教学&#xff0c;擅长Java、Python、微信小程序、Golang、安卓Android等项目实战。接项目定制开发、代码讲解、答辩教学、文档编写、降重等。 ☑文末获取源码☑ 精彩专栏推荐⬇⬇⬇ Java项目 Py…

基于微信小程序的宠物之家的设计与实现

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、SSM项目源码 系统展示 基于微信小程序JavaSpringBootVueMySQL的宠物之家/宠物综合…

feign client发送Post请求,发送对象参数,服务端接收不到正确参数报错排查

记一次feignclient发送请求服务端接收不到正确参数排查 服务端代码&#xff1a; Operation(summary "Create team")PostMapping("post")RequiresPermissions("team:add")public RestResponse addTeam(Valid Team team) {this.teamService.crea…

『功能项目』切换职业技能面板【49】

我们打开上一篇48切换职业面板的项目&#xff0c; 本章要做的事情是制作第二职业法师技能面板、第三职业面板并且完成切换 双击打开Canvas进入预制体空间 复制三个技能栏面板 重命名 设置第一技能栏 设置第二职业技能栏 设置第三职业技能栏 修改脚本&#xff1a;ChangeProfess…

经纬恒润高压电池管理系统,助力新能源汽车飞速发展

随着新能源汽车行业的快速发展&#xff0c;电池管理系统作为关键技术之一&#xff0c;其重要性日益凸显。经纬恒润自主研发的高压电池管理系统&#xff08;Battery Management System&#xff0c;BMS&#xff09;&#xff0c;凭借卓越的性能与先进的技术&#xff0c;在新能源汽…

Mac清理其他文件:释放存储空间的高效指南

每个Mac用户都可能遇到存储空间不足的问题&#xff0c;尤其是当“其他”文件积累到一定体积时。在Mac上&#xff0c;“其他”文件通常包括各种系统文件、缓存、文档以及不被归类为应用程序、照片、电影或音乐的其他类型的文件。这些文件往往不易被注意&#xff0c;但逐渐占用了…

量化交易的个人见解

程序化交易在国内兴起有些年数了&#xff0c;个人以为&#xff0c;程序化交易与量化投资的关系&#xff0c;在于两者侧重点有差别。程序化交易侧重于下单的动作是机器自动执行的&#xff0c;量化投资则侧重于投资分析的过程是通过一个量化模型来实现的&#xff0c;所以量化投资…