首页 > 编程语言 >Java设计模式之责任链模式详细讲解和案例示范

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

时间:2024-09-12 18:20:54浏览次数:3  
标签:Java 请求 处理 模式 责任 讲解 过滤器 设计模式 order

在本文中,我们将详细讲解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 {
    @Override
    public void handleOrder(Order order) {
        if (order.getStatus().equals("PAID")) {
            // 处理已支付的订单
        } else if (nextHandler != null) {
            nextHandler.handleOrder(order);
        }
    }
}

class ShippedOrderHandler extends OrderHandler {
    @Override
    public void handleOrder(Order order) {
        if (order.getStatus().equals("SHIPPED")) {
            // 处理已发货的订单
        } else if (nextHandler != null) {
            nextHandler.handleOrder(order);
        }
    }
}

class DeliveredOrderHandler extends OrderHandler {
    @Override
    public 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 {
    @Override
    public 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 {

    @Override
    protected void doFilterInternal(HttpServletRequest request,
                                    HttpServletResponse response,
                                    FilterChain filterChain)
            throws ServletException, IOException {

        String token = request.getHeader("Authorization");

        // 简单的认证逻辑:如果存在token,就通过认证
        if (token != null && token.startsWith("Bearer ")) {
            // 模拟将用户信息存入SecurityContext
            SecurityContextHolder.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 {

    @Override
    protected 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. 总结

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

标签:Java,请求,处理,模式,责任,讲解,过滤器,设计模式,order
From: https://blog.csdn.net/weixin_39996520/article/details/142152936

相关文章

  • Java中包装类的学习
    包装类目录包装类什么是包装类包装类的特点基本数据类型和包装类基本数据类型包装类区别和联系包装类的共同特点自动拆箱/装箱什么是包装类java中的数据类型int,double等不是对象,无法通过向上转型获取到Object提供的方法,而像String却可以,只因为String是一个对象而不是一个类型......
  • Java将Word文档转换为PDF文件常用方法总结
    1.使用ApachePOI+iTextApachePOI是一个流行的Java库,用于处理MicrosoftOffice文档。可以使用它来读取Word文档,而iText可以用来生成PDF文件。组合这两个库可以实现Word到PDF的转换。示例代码importorg.apache.poi.xwpf.usermodel.XWPFDocument;importorg.apach......
  • 基于Java的共享经济背景下校园闲置物品交易平台(2024最新,原创项目)
    文章目录1.前言2.系统演示录像3.论文参考4.代码运行展示图5.技术框架5.1SpringBoot技术介绍5.2Vue技术介绍6.可行性分析7.系统测试7.1系统测试的目的7.2系统功能测试8.数据库表设计9.代码参考10.数据库脚本11.找我做程序,有什么保障?12.联系我们1.前......
  • Java类的组成
    类的组成属性在类中通过成员变量来体现。行为在类中通过成员方法来体现。类如何定义publicclass类名称{成员属性成员方法}例如一个手机类点击查看代码publicclassPhone{//成员属性Stringbrand;//手机品牌doubleprice;//手机价格//成......
  • Java Executors类的9种创建线程池的方法及应用场景分析
    在Java中,Executors类提供了多种静态工厂方法来创建不同类型的线程池。在学习线程池的过程中,一定避不开Executors类,掌握这个类的使用、原理、使用场景,对于实际项目开发时,运用自如,以下是一些常用的方法,一一细说:newCachedThreadPool():创建一个可缓存的线程池,如果线程池中的......
  • 基于javassm小区社区物业管理系统的计算机毕设
    小区物业管理系统的设计与实现摘 要随着我国市场经济的快速发展和人们生活水平的不断提高,简单的小区服务已经不能满足人们的需求。如何利用先进的管理手段,提高业主管理水平,是当今社会所面临的一个重要课题。要想提高小区物业管理水平,必须全方位地提高业主管理意识。小区物业管理......
  • 解决Java8中使用LocalDate接收参数异常 not supported by default
    背景在spring项目中经常遇到需要传参为时间类型的参数,java8提供了很方便的日期:LocalDate时间:LocalTime及LocalDateTime类型,在spring接口中直接使用这些类型接收入参时会报错Java8date/timetypejava.time.LocalDatenotsupportedbydefault原因是jackson库默认......
  • 计算机专业毕设推荐-基于Java的个人健康运动饮食记录小程序
    精彩专栏推荐订阅:在下方专栏......
  • Java面向对象知识全解(OOP)
    类是抽象的,概念的;对象是具体的,实际的;类说对象的模板,对象是类的一个个体属性属性的定义语法同变量:访问修饰符属性类型属性名属性的定义类型可以是任意类型,包含基本类型或引用类型属性如果不赋值,有默认值,规则和数组一样创建对象方法先声明再创建Catcat;cat=newCat(......
  • 基于javaC语言试题生成与考试系统的计算机毕设
    C语言试题生成与考试系统摘 要当前,网络教学方兴未艾。网上考试已在其中扮演了重要的角色,传统试卷考试方式有待提高。网络教学已从其规范性、科学性及考试工作组织、管理的统一性,影响到教学质量的好坏。基于此,本系统开发实现了基于B/S模式的c试题生成与考试系统,其中数据库采用MYSQ......