策略模式在spring哪里用到了?

策略模式是 Spring 框架中应用得最为广泛的设计模式之一,几乎贯穿了整个框架的设计。它的核心思想是定义一系列的算法,将它们一个个封装起来,并且使它们可以相互替换,从而让算法的变化独立于使用算法的客户端。

在 Spring 中,这通常体现为:

  1. 定义一个策略接口
  2. 提供多个不同的接口实现类(即各种策略)。
  3. 在运行时,根据条件(如配置、上下文环境等)由 Spring 容器动态地决定使用哪个具体的实现(通常通过依赖注入实现)。

1. Resource 接口与资源加载

这是最直观的例子之一。

  • 策略接口org.springframework.core.io.Resource
  • 具体策略
    • UrlResource:用于加载网络资源。
    • ClassPathResource:用于加载类路径下的资源。
    • FileSystemResource:用于加载文件系统资源。
    • ServletContextResource:用于加载 Web 应用上下文中的资源。
  • 策略选择者org.springframework.core.io.ResourceLoader(其默认实现是 DefaultResourceLoader

如何使用

你不需要直接 new 一个具体的 Resource 实现,而是通过 ResourceLoader 来获取。ResourceLoader 会根据你传入的资源字符串的前缀(如 classpath:file:http:)自动选择正确的策略。

java 复制代码
ResourceLoader resourceLoader = new DefaultResourceLoader();
Resource resource = resourceLoader.getResource("classpath:config.xml");
// 此时 resource 的实际类型是 ClassPathResource

Resource httpResource = resourceLoader.getResource("https://example.com/data.json");
// 此时 httpResource 的实际类型是 UrlResource

2. Bean 实例化策略

Spring 需要根据类定义创建 Bean 的实例,而创建实例的方式有多种。

  • 策略接口org.springframework.beans.factory.support.InstantiationStrategy
  • 具体策略
    • SimpleInstantiationStrategy:使用普通的反射调用构造函数来实例化对象。
    • CglibSubclassingInstantiationStrategy:使用 CGLIB 动态生成子类的方式来进行实例化(例如,需要方法注入或代理没有默认构造函数的类时)。它是 SimpleInstantiationStrategy 的子类。
  • 策略选择者AbstractAutowireCapableBeanFactorycreateBeanInstance() 方法中会根据 Bean 的定义(是否需要方法注入、是否有 lookup-method 等)来决定使用哪种实例化策略。

3. HandlerMapping

在 Spring MVC 中,需要根据 HTTP 请求找到对应的处理控制器(Controller)。

  • 策略接口org.springframework.web.servlet.HandlerMapping
  • 具体策略
    • RequestMappingHandlerMapping:基于 @RequestMapping 注解将请求映射到方法(目前最主流的策略)。
    • BeanNameUrlHandlerMapping:根据 Bean 的名字(以 "/" 开头)来映射 URL。
    • SimpleUrlHandlerMapping:通过配置文件手动配置 URL 和 Controller 的映射关系。
  • 策略选择者DispatcherServlet 会持有多个 HandlerMapping 组件,形成一个责任链。在处理请求时,它会按顺序遍历这些策略,直到某一个策略能返回一个可用的 Handler。

4. HandlerAdapter

同样在 Spring MVC 中,DispatcherServlet 拿到 Handler 之后,需要调用这个 Handler 的真正处理方法。但由于 Handler 的形式多种多样(如普通的 @Controller、传统的 Controller 接口实现、HttpRequestHandler 等),调用方式也不同。

  • 策略接口org.springframework.web.servlet.HandlerAdapter
  • 具体策略
    • RequestMappingHandlerAdapter:用于适配和执行基于 @RequestMapping 注解的方法。
    • SimpleControllerHandlerAdapter:用于适配实现了 Controller 接口的处理器。
    • HttpRequestHandlerAdapter:用于适配实现了 HttpRequestHandler 接口的处理器。
  • 策略选择者DispatcherServlet 会遍历所有 HandlerAdapter,询问其 supports() 方法是否支持当前找到的 Handler。找到第一个支持的 Adapter 后,就使用它来执行目标方法。
java 复制代码
// 在 DispatcherServlet 中的核心逻辑
protected HandlerAdapter getHandlerAdapter(Object handler) throws ServletException {
    for (HandlerAdapter adapter : this.handlerAdapters) {
        if (adapter.supports(handler)) { // 询问每个策略:你能处理这个Handler吗?
            return adapter; // 找到能处理的策略
        }
    }
    ...
}

5. PlatformTransactionManager(平台事务管理器)

这是一个业务层面非常经典的策略模式应用。

  • 策略接口org.springframework.transaction.PlatformTransactionManager
  • 具体策略
    • DataSourceTransactionManager:用于管理 JDBC 本地事务。
    • JpaTransactionManager:用于管理 JPA 事务。
    • JtaTransactionManager:用于管理分布式 JTA 事务。
    • HibernateTransactionManager:用于管理 Hibernate 事务。
  • 策略选择者 :应用程序开发者。我们通过在 Spring 配置文件中配置具体的实现(策略),然后在代码中通过 @Transactional 注解使用。Spring 会根据配置注入正确的策略来管理事务。
xml 复制代码
<!-- 配置具体策略 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <property name="dataSource" ref="dataSource"/>
</bean>
java 复制代码
// 使用统一接口,无需关心具体策略
@Transactional
public void performBusinessLogic() {
    // ...
}

相关推荐
青衫码上行1 小时前
【Java Web学习 | 第14篇】JavaScript(8) -正则表达式
java·前端·javascript·学习·正则表达式
拽着尾巴的鱼儿2 小时前
工具篇:Window10 增加虚拟内存&Idea项目启动内存配置
java·ide·intellij-idea
q***13612 小时前
SpringSecurity相关jar包的介绍
java·jar
皮影w3 小时前
Java SpringAOP入门
java·开发语言
007php0073 小时前
Redis面试题解析:Redis的数据过期策略
java·网络·redis·缓存·面试·职场和发展·php
w***48823 小时前
Spring Boot3.x集成Flowable7.x(一)Spring Boot集成与设计、部署、发起、完成简单流程
java·spring boot·后端
u***1373 小时前
详解tomcat中的jmx监控
java·tomcat
Vic101013 小时前
Java 序列化与反序列化:深入解析与实践
java·开发语言
后端小张3 小时前
【JAVA 进阶】Spring Cloud 微服务全栈实践:从认知到落地
java·开发语言·spring boot·spring·spring cloud·微服务·原理