当前位置: 首页 > news >正文

Spring 策略模式实现

Spring 策略模式实现:工厂方法与自动注入详解

1. 背景介绍

在复杂的业务系统中,我们常常需要根据不同的场景选择不同的处理策略。本文将详细介绍在 Spring 框架中实现策略模式的两种主要方法。

2. 方案一: 手动注册工厂模式

2.1 定义工厂类

@Component
public class CalculateHandlerFactory implements InitializingBean, ApplicationContextAware {private ApplicationContext applicationContext;// 存储不同策略的映射private static final Map<String, CalculateHandler> HANDLER_MAP = new HashMap<>();// 初始化方法,自动注册所有处理器@Overridepublic void afterPropertiesSet() throws Exception {applicationContext.getBeansOfType(CalculateHandler.class).values().forEach(v -> HANDLER_MAP.putIfAbsent(v.getType(), v));}// 设置 ApplicationContext@Overridepublic void setApplicationContext(ApplicationContext applicationContext) throws BeansException {this.applicationContext = applicationContext;}// 根据类型获取对应的处理器public CalculateHandler getCalculateHandler(String calculateType) {return HANDLER_MAP.getOrDefault(calculateType, null);}
}

2.2 实现步骤

  1. 实现 InitializingBean 接口,在 Bean 初始化时自动注册处理器
  2. 实现 ApplicationContextAware 接口,获取 Spring 应用上下文
  3. 使用 Map 存储不同类型的处理器
  4. 提供根据类型获取处理器的方法

2.3 优点

  • 对注册过程有精细化控制
  • 可以添加自定义的注册逻辑
  • 灵活性高

3. 方案二: 自动注入策略模式

3.1 定义策略接口

public interface CalculateHandler {// 获取处理类型String getType();// 具体计算方法double calculate(double a, double b);
}

3.2 实现具体策略

@Component
public class AddCalculateHandler implements CalculateHandler {@Overridepublic String getType() {return "add";}@Overridepublic double calculate(double a, double b) {return a + b;}
}@Component
public class SubtractCalculateHandler implements CalculateHandler {@Overridepublic String getType() {return "subtract";}@Overridepublic double calculate(double a, double b) {return a - b;}
}

3.3 策略管理服务

@Service
public class CalculateService {@Autowiredprivate Map<String, CalculateHandler> calculateHandlerMap;public double calculate(String type, double a, double b) {CalculateHandler handler = calculateHandlerMap.get(type);if (handler == null) {throw new IllegalArgumentException("未找到对应的计算处理器");}return handler.calculate(a, b);}
}

3.4 工作原理详解

3.4.1 自动注册机制

当使用 @Autowired Map<String, Interface> 时,Spring 会:

  • 扫描所有实现指定接口的 Bean
  • 使用 Bean 名称作为 Map 的 Key
  • 使用 Bean 实例作为 Map 的 Value
3.4.2 Bean 名称规则
  • 默认使用类名首字母小写作为 Bean 名称
  • 可以通过 @Component("customName") 自定义 Bean 名称

4. 高级用法

4.1 按优先级排序

@Service
public class CalculateService {@Autowiredprivate Map<String, CalculateHandler> calculateHandlerMap;// 按照 @Order 注解排序@Autowiredprivate List<CalculateHandler> calculateHandlerList;
}

4.2 自定义 Key 获取

@Service
public class CalculateService {@Autowiredprivate Map<String, CalculateHandler> calculateHandlerMap;// 使用自定义方法获取 Keypublic double calculate(String type, double a, double b) {CalculateHandler handler = calculateHandlerMap.values().stream().filter(h -> h.getType().equals(type)).findFirst().orElseThrow(() -> new IllegalArgumentException("未找到处理器"));return handler.calculate(a, b);}
}

5. 两种方案对比

特性手动注册自动注入
代码复杂度较高较低
灵活性
初始化控制精细简单
性能略低略高

6. 优点和适用场景

6.1 优点

  1. 代码解耦
  2. 动态扩展
  3. 无需手动维护注册表
  4. 充分利用 Spring 依赖注入机制

6.2 适用场景

  • 策略模式
  • 插件化开发
  • 可插拔的业务处理器
  • 系统扩展性要求高的场景

7. 注意事项

  1. 保证 Bean 名称唯一性
  2. 接口设计要合理
  3. 做好异常处理
  4. 考虑性能和扩展性

8. 最佳实践

  1. 保持接口简洁明了
  2. 明确定义每个策略的职责
  3. 合理设计方法签名
  4. 添加必要的异常处理
  5. 考虑性能和扩展性

结论

通过 Spring 的依赖注入和自动装配机制,我们可以非常优雅地实现策略模式,使代码更加灵活、可读和可维护。选择合适的实现方式,需要根据具体的业务场景和系统架构来权衡。

http://www.dtcms.com/a/297149.html

相关文章:

  • 配置Mac/Linux终端启动执行脚本
  • 电子电子架构 --- 软件项目的开端:裁剪
  • 现代 C++ 开发工作流(VSCode / Cursor)
  • ubuntu/centos系统ping 不通域名的解决方案
  • 清理DNS缓存
  • Ubuntu 环境下创建并启动一个 MediaMTX 的 systemd 服务
  • 缓存HDC内容用于后续Direct2D绘制.
  • 数据仓库深度探索系列 | 开篇:开启数仓建设新征程
  • 验证回文串-leetcode
  • Nginx简单介绍
  • 【STM32】Keil + FreeRTOS + HAL DMA + UART 空闲中断 接收异常
  • 【矩阵专题】Leetcode48.旋转图像(Hot100)
  • leetcode_122 买卖股票的最佳时机II
  • STM32与ADS1220实现多通道数据采集的完整分析和源程序
  • Comfyui中Upscale Image By 几种放大方法的区别
  • Java研学-RabbitMQ(三)
  • Centos7安装rabbitmq
  • RabbitMQ—HAProxy负载均衡
  • React性能优化终极指南:memo、useCallback、useMemo全解析
  • Ubuntu22 上,用C++ gSoap 创建一个简单的webservice
  • NineData 数据库 DevOps 全面支持 GaussDB,国产化管理再升级!
  • Spring Boot 自动装配底层源码实现详解
  • 国产DevOps平台Gitee:如何重塑中国企业研发效能新格局
  • Java 单元测试详解:从入门到实战,彻底掌握 JUnit 5 + Mockito + Spring Boot 测试技巧
  • react中 多个层级 组件数据同用 组件之间传值 usecontext useReducer
  • Gitee如何成为国内企业DevOps转型的首选平台?
  • 璞致 PZSDR-P101:ZYNQ7100+AD9361 架构软件无线电平台,重塑宽频信号处理范式
  • ERNIE-4.5-0.3B 实战指南:文心一言 4.5 开源模型的轻量化部署与效能跃升
  • 规则分配脚本
  • 初识JVM--从Java文件到机器指令