策略模式在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 小时前
从 ForkJoinPool 的 Compensate 看并发框架的线程补偿思想
java·后端·源码阅读
躺平大鹅3 小时前
Java面向对象入门(类与对象,新手秒懂)
java
初次攀爬者4 小时前
RocketMQ在Spring Boot上的基础使用
java·spring boot·rocketmq
花花无缺4 小时前
搞懂@Autowired 与@Resuorce
java·spring boot·后端
Derek_Smart5 小时前
从一次 OOM 事故说起:打造生产级的 JVM 健康检查组件
java·jvm·spring boot
NE_STOP6 小时前
MyBatis-mybatis入门与增删改查
java
孟陬9 小时前
国外技术周刊 #1:Paul Graham 重新分享最受欢迎的文章《创作者的品味》、本周被划线最多 YouTube《如何在 19 分钟内学会 AI》、为何我不
java·前端·后端
想用offer打牌9 小时前
一站式了解四种限流算法
java·后端·go
华仔啊10 小时前
Java 开发千万别给布尔变量加 is 前缀!很容易背锅
java