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

在SpringBoot项目中策略模式的使用

使用策略模式之前的代码

 @Overridepublic void updateExam(String id, ExamUpdateDTO examUpdateDTO) {logger.info("Service: 修改考试场次, ID: {}, 数据: {}", id, examUpdateDTO);Exam existingExam = mongoDBUtils.findById(id, Exam.class);if (existingExam == null) {throw new EntityNotFoundException("考试场次不存在: " + id);}Map<String, Object> examUpdateMap = new HashMap<>();if (examUpdateDTO.getExamName() != null) {examUpdateMap.put("examName", examUpdateDTO.getExamName());}if (examUpdateDTO.getExamDescription() != null) {examUpdateMap.put("examDescription", examUpdateDTO.getExamDescription());}if (examUpdateDTO.getStartTime() != null) {examUpdateMap.put("startTime", examUpdateDTO.getStartTime());}if (examUpdateDTO.getDuration() != null) {examUpdateMap.put("duration", examUpdateDTO.getDuration());}if (examUpdateDTO.getStatus() != null) {examUpdateMap.put("status", examUpdateDTO.getStatus());}if (examUpdateMap.isEmpty()) {logger.warn("Service: 没有可更新的考试场次字段, ID: {}", id);return;}examUpdateMap.put("updateTime", System.currentTimeMillis());boolean examUpdated = mongoDBUtils.updateOne(Criteria.where("ID").is(id), examUpdateMap, Exam.class);if (!examUpdated) {logger.warn("Service: 更新考试场次失败 or no changes made, ID: {}", id);}logger.info("Service: 考试场次基础信息更新已提交, ID: {}", id);boolean needsStudentUpdate = examUpdateDTO.getStartTime() != null || examUpdateDTO.getDuration() != null;if (needsStudentUpdate) {Map<String, Object> studentUpdateMap = new HashMap<>();if (examUpdateDTO.getStartTime() != null) {studentUpdateMap.put("startTime", examUpdateDTO.getStartTime());}if (examUpdateDTO.getDuration() != null) {studentUpdateMap.put("duration", examUpdateDTO.getDuration());}if (!studentUpdateMap.isEmpty()) {Criteria studentCriteria = Criteria.where("examID").is(id);boolean studentsUpdated = mongoDBUtils.updateMany(studentCriteria, studentUpdateMap, CompetitionSignUpLog.class);if (studentsUpdated) {logger.info("Service: 成功级联更新考试场次 ID {} 下学员的 startTime/duration.", id);} else {logger.warn("Service: 级联更新考试场次 ID {} 下学员的 startTime/duration 操作未被确认或无匹配学员.", id);}}}}

创建策略接口即策略实现类

/*** 更新考试场次信息的策略接口*/
public interface UpdateExamStrategy {/*** 根据DTO将待更新的字段填充到相应的map中。** @param existingExam   当前的考试实体 (可用于比较或特定逻辑)* @param dto            考试更新DTO* @param examUpdateMap  用于收集考试实体自身更新的字段* @param studentUpdateMap 用于收集需要级联更新到学员报名记录的字段*/void applyUpdate(Exam existingExam, ExamUpdateDTO dto, Map<String, Object> examUpdateMap, Map<String, Object> studentUpdateMap);
} 
@Component
public class ExamDescriptionUpdateStrategy implements UpdateExamStrategy {@Overridepublic void applyUpdate(Exam existingExam, ExamUpdateDTO dto, Map<String, Object> examUpdateMap, Map<String, Object> studentUpdateMap) {if (dto.getExamDescription() != null) {examUpdateMap.put("examDescription", dto.getExamDescription());}}
}
@Component
public class ExamDurationUpdateStrategy implements UpdateExamStrategy {@Overridepublic void applyUpdate(Exam existingExam, ExamUpdateDTO dto, Map<String, Object> examUpdateMap, Map<String, Object> studentUpdateMap) {if (dto.getDuration() != null) {examUpdateMap.put("duration", dto.getDuration());studentUpdateMap.put("duration", dto.getDuration()); // Also update for students}}
} ``````java
@Component
public class ExamNameUpdateStrategy implements UpdateExamStrategy {@Overridepublic void applyUpdate(Exam existingExam, ExamUpdateDTO dto, Map<String, Object> examUpdateMap, Map<String, Object> studentUpdateMap) {if (dto.getExamName() != null) {examUpdateMap.put("examName", dto.getExamName());}}
} 
@Component
public class ExamStartTimeUpdateStrategy implements UpdateExamStrategy {@Overridepublic void applyUpdate(Exam existingExam, ExamUpdateDTO dto, Map<String, Object> examUpdateMap, Map<String, Object> studentUpdateMap) {if (dto.getStartTime() != null) {examUpdateMap.put("startTime", dto.getStartTime());studentUpdateMap.put("startTime", dto.getStartTime()); // Also update for students}}
} 
@Component
public class ExamStatusUpdateStrategy implements UpdateExamStrategy {@Overridepublic void applyUpdate(Exam existingExam, ExamUpdateDTO dto, Map<String, Object> examUpdateMap, Map<String, Object> studentUpdateMap) {if (dto.getStatus() != null) {examUpdateMap.put("status", dto.getStatus());}}
}

使用策略模式之后的updateExam方法

@Service
public class ExamServiceImpl implements ExamService {private static final Logger logger = LogManager.getLogger(ExamServiceImpl.class);@Autowiredprivate MongoDBUtils mongoDBUtils;private final List<UpdateExamStrategy> updateExamStrategies;@Autowiredpublic ExamServiceImpl(MongoDBUtils mongoDBUtils, List<UpdateExamStrategy> updateExamStrategies) {this.mongoDBUtils = mongoDBUtils; // Ensure mongoDBUtils is assigned if it's not already via @Autowired field injectionthis.updateExamStrategies = updateExamStrategies;}//...省略其他方法的实现@Overridepublic void updateExam(String id, ExamUpdateDTO examUpdateDTO) {logger.info("Service: 修改考试场次, ID: {}, 数据: {}", id, examUpdateDTO);Exam existingExam = mongoDBUtils.findById(id, Exam.class);if (existingExam == null) {throw new EntityNotFoundException("考试场次不存在: " + id);}Map<String, Object> examUpdateMap = new HashMap<>();Map<String, Object> studentUpdateMap = new HashMap<>();// 遍历所有策略并应用更新for (UpdateExamStrategy strategy : updateExamStrategies) {strategy.applyUpdate(existingExam, examUpdateDTO, examUpdateMap, studentUpdateMap);}if (!examUpdateMap.isEmpty()) {examUpdateMap.put("updateTime", System.currentTimeMillis());boolean examUpdated = mongoDBUtils.updateOne(Criteria.where("ID").is(id), examUpdateMap, Exam.class);if (!examUpdated) {logger.warn("Service: 更新考试场次失败 or no changes made, ID: {}", id);} else {logger.info("Service: 考试场次基础信息更新已提交, ID: {}", id);}} else {logger.info("Service: 没有可更新的考试场次字段, ID: {}", id);// 如果 examUpdateMap 为空,但 studentUpdateMap 不为空,则不需要返回// 即使考试本身没有变化,关联的学生信息可能仍需更新}if (!studentUpdateMap.isEmpty()) {Criteria studentCriteria = Criteria.where("examID").is(id);boolean studentsUpdated = mongoDBUtils.updateMany(studentCriteria, studentUpdateMap, CompetitionSignUpLog.class);if (studentsUpdated) {logger.info("Service: 成功级联更新考试场次 ID {} 下学员的相关信息.", id);} else {logger.warn("Service: 级联更新考试场次 ID {} 下学员信息的操作未被确认或无匹配学员.", id);}}}}   

在这个场景中,策略的选择是由Spring的依赖注入机制自动完成的。具体工作原理:

  1. 策略的收集
  • 由于每个策略类(如ExamDescriptionUpdateStrategyExamStatusUpdateStrategyExamStartTimeUpdateStrategy)都标注了@Component注解
  • Spring会自动扫描并实例化这些实现了UpdateExamStrategy接口的类
  • 所有策略实例会被自动注入到ExamServiceImpl中的updateExamStrategies列表中
  1. 策略的顺序
  • Spring在注入时会按照以下规则确定策略在列表中的顺序:
    • 默认按照类名的字母顺序
    • 可以通过@Order注解或实现Ordered接口来自定义顺序
  1. 策略的使用
@Autowired
public ExamServiceImpl(MongoDBUtils mongoDBUtils, List<UpdateExamStrategy> updateExamStrategies) {this.mongoDBUtils = mongoDBUtils;this.updateExamStrategies = updateExamStrategies;  // Spring自动注入所有策略
}
  1. 遍历执行
for (UpdateExamStrategy strategy : updateExamStrategies) {strategy.applyUpdate(existingExam, examUpdateDTO, examUpdateMap, studentUpdateMap);
}
  • 每次遍历会取出列表中的一个策略实例
  • 每个策略实例都是一个具体的策略类的对象(如ExamDescriptionUpdateStrategy的实例)
  • 调用策略的applyUpdate方法时,会执行对应策略类中的具体实现

这是一个典型的策略模式结合Spring依赖注入的应用,它的好处是:

  • 解耦:各个更新策略相互独立
  • 可扩展:添加新的更新策略只需创建新的策略类并添加@Component注解
  • 维护性好:每个策略类负责处理特定的更新逻辑

总结

这里添加了一个构造函数来注入 mongoDBUtils 和
updateExamStrategies。如果你的项目中 mongoDBUtils 已经通过字段上的 @Autowired
注入,那么在构造函数中再次赋值是多余的,但通常无害。Spring 的依赖注入会处理好这些。
主要逻辑已更改为: 初始化 examUpdateMap 和 studentUpdateMap。 遍历所有注入的 UpdateExamStrategy 策略。 每个策略根据 ExamUpdateDTO 的内容,决定是否要向 examUpdateMap(用于更新 Exam 实体本身)或
studentUpdateMap(用于级联更新 CompetitionSignUpLog 实体)添加键值对。 如果
examUpdateMap 不为空,则更新 Exam 实体。 如果 studentUpdateMap 不为空,则更新相关的CompetitionSignUpLog 实体。 这种方式将每个字段的更新逻辑(包括是否需要级联更新)封装在各自的策略类中,使得 updateExam 方法本身更加简洁,并且易于扩展新的更新逻辑。

相关文章:

在SpringBoot项目中策略模式的使用

使用策略模式之前的代码 Overridepublic void updateExam(String id, ExamUpdateDTO examUpdateDTO) {logger.info("Service: 修改考试场次, ID: {}, 数据: {}", id, examUpdateDTO);Exam existingExam mongoDBUtils.findById(id, Exam.class);if (existingExam nu…...

在 Docker 中启动 Jupyter Notebook

文章目录 一、创建容器二、Conda安装三、安装 Jupyter四、启动 Jupyter五、注册内核来使用虚拟环境小结 一、创建容器 可以先查看宿主机8888端口是否被占用&#xff0c;无输出&#xff0c;表明端口未被任何进程占用&#xff0c;如果有LISTEN&#xff0c;可能在创建容器的时候需…...

IP 地址反向解析(IP反查域名)原理与应用

一、IP 地址反向解析的原理与技术细节 IP 地址反向解析&#xff08;Reverse IP Lookup&#xff09;是一种将 IP 地址映射回其关联域名或主机名的网络技术&#xff0c;与常见的正向 DNS 解析&#xff08;将域名解析为 IP 地址&#xff09;形成互补。这一过程在网络安全研究、漏…...

CodeTop之LRU缓存

题目链接 146. LRU 缓存 - 力扣&#xff08;LeetCode&#xff09; 题目解析 算法原理 我们使用双向链表哈希表的形式来模拟缓存机制 首先我们要自己实现一个双链表, 自己写一个内部类, 这个内部类记录了key,value,prev,next(前驱和后继), 后续我们就通过这个内部类来构造双…...

uboot常用命令之eMMC/SD卡命令

eMMC和SD卡(TF卡)是同一类设备&#xff0c;以下命令二者是通用&#xff0c;本章节主要以eMMC举例说明命令的使用。 使用help mmc可以看到mmc相关命令列表以及其对应命令用法&#xff1a; > help mmc 一、mmc dev 使用mmc list可以看到当前系统挂载的所有mmc设备&#xff…...

【Kafka】编写消费者开发模式时遇到‘未解析的引用‘SIGUSR1’’

在编写消费者开发模式时&#xff0c;不要用简单的consumer&#xff0c;会导致消费数据不全的情况&#xff0c;需要用ConsumerGroup。 代码可以参考官方实例&#xff1a;https://github.com/Shopify/sarama/tree/main/examples/consumergroup 问题描述&#xff1a; 编写消费者开…...

DeepSeek 赋能教育游戏化:AI 重构学习体验的技术密码

目录 一、引言&#xff1a;教育游戏化与 DeepSeek 的相遇二、DeepSeek 技术剖析2.1 核心架构2.2 关键技术 三、教育游戏化设计的奥秘3.1 概念与意义3.2 常见方法与元素3.3 成功案例借鉴 四、DeepSeek 在教育游戏化设计中的多面应用4.1 个性化学习路径打造4.2 智能教学辅助工具4…...

Docker run命令-p参数详解

端口映射基础语法 docker run -p <宿主机端口>:<容器端口> 操作示例 docker run -d --restartalways --namespug -p 5000:80 registry.aliyuncs.com/openspug/spug参数解析 -d&#xff1a;后台运行容器--restartalways&#xff1a;设置容器自动重启--namespug&…...

知识宇宙-学习篇:学编程为什么从C语言开始学起?

名人说&#xff1a;博观而约取&#xff0c;厚积而薄发。——苏轼《稼说送张琥》 创作者&#xff1a;Code_流苏(CSDN)&#xff08;一个喜欢古诗词和编程的Coder&#x1f60a;&#xff09; 目录 一、C语言的历史地位与影响力1. 编程语言的"鼻祖"2. 现代技术的基础 二、…...

Mybatis-入门程序、 数据库连接池、XML映射配置文件、MybatisX

一. Mybatis 1. Mybatis是一款优秀的持久层框架&#xff0c;用于简化jdbc的开发 2. Mybatis本是Apache的一个开源项目iBatis&#xff0c;2010年这个项目有Apache迁移到了Google code&#xff0c;并且改名为MyBatis&#xff0c;2013年11月迁移到Github 3.官网&#xff1a;MyBat…...

互联网大厂Java求职面试:Spring Cloud微服务架构设计中的挑战与解决方案

互联网大厂Java求职面试&#xff1a;Spring Cloud微服务架构设计中的挑战与解决方案 面试场景设定 郑薪苦是一位拥有丰富实战经验的Java开发者&#xff0c;他正在参加一场由某知名互联网大厂的技术总监主持的面试。这场面试将围绕Spring Cloud微服务架构展开&#xff0c;涵盖…...

BUUCTF [ZJCTF 2019]EasyHeap

前置知识点: unlink知识点和手法-CSDN博客 [ZJCTF 2019]EasyHeap [ZJCTF 2019]EasyHeap 1.准备 2.ida分析 main函数 int __fastcall __noreturn main(int argc, const char **argv, const char **envp) {int n3; // eaxchar buf[8]; // [rsp0h] [rbp-10h] BYREFunsigned …...

机器学习AI精准预测复合材料性能、材料结构设计优化;数据驱动加速新材料研发,百年难遇的组合打破科研壁垒!

在人工智能与复合材料技术融合的背景下&#xff0c;复合材料的研究和应用正迅速发展&#xff0c;创新解决方案层出不穷。从复合材料性能的精确预测到复杂材料结构的智能设计&#xff0c;从数据驱动的材料结构优化到多尺度分析&#xff0c;人工智能技术正以其强大的数据处理能力…...

apache http client连接池实现原理

在java开发中我们经常会涉及到http 请求接口&#xff0c;一般有几种方式&#xff1a; java自带的 HttpURLConnectionokHttpClientapache http client 一般我们使用apache http client会比较多点&#xff0c;在代码中会进行如下调用方式&#xff1a; private static class Htt…...

如何做好一份网络安全技术文档?

在网络安全领域&#xff0c;技术文档是沟通、记录和分享专业知识的桥梁。它不仅帮助团队成员理解系统设计和安全策略&#xff0c;也为未来的维护和更新提供了宝贵的参考。对于编写网络安全技术文档来说&#xff0c;结构清晰、内容准确以及易于理解是至关重要的。本文将介绍如何…...

Android Studio 介绍

如何关闭或彻底删除一个工程 基于Android Studio的android入门——如何关闭或彻底删除一个工程 搜索内容 Android Studio高效指南&#xff1a;快速查找技巧大揭秘 build命令&#xff1a;gradle app:assembleDebug 命令解析 1. 命令结构与作用 核心功能&#xff1a;该命令…...

MD5加密(Java)

首先来看数据库里的一张员工信息表&#xff1a; 问题&#xff1a; 员工表中的密码是明文存储&#xff0c;安全性太低。 解决思路&#xff1a; 将明文密码加密后存储&#xff0c;提高安全性。 加密方式有很多&#xff0c;这里简单介绍 MD5加密方式 &#xff1a; (详细解释请转…...

[攻防世界] easyphp writeup

知识点 科学计数法的妙用 9e9 指定结尾MD5值的爆破array_search() 函数用于在数组中搜索某个值&#xff0c;并返回对应的键名。如果找不到该值&#xff0c;则返回 false 默认值匹配&#xff1a;可以利用整数绕过字符串匹配机制stricttrue时&#xff0c;数据类型和值都需要匹配…...

力扣热题100之LRU缓存机制

题目 请你设计并实现一个满足 LRU (最近最少使用) 缓存 约束的数据结构。 实现 LRUCache 类&#xff1a; LRUCache(int capacity) 以 正整数 作为容量 capacity 初始化 LRU 缓存 int get(int key) 如果关键字 key 存在于缓存中&#xff0c;则返回关键字的值&#xff0c;否则返…...

如何不规范的设置密码

上来就干 当我们使用服务器的时候&#xff0c;有时候需要一些非常简单的密码&#xff0c;来方便使用&#xff0c;但是自己完全可控的环境下&#xff0c;我们希望我们的密码足够的简单&#xff0c;比如&#xff0c;可能它的密码就是123&#xff0c;或者是1&#xff1f; 但是当你…...

数据安全与纵深访问控制:构建数字时代的安全防线

在当今数字经济蓬勃发展的时代&#xff0c;数据已成为与土地、劳动力、资本同等重要的生产要素&#xff0c;被誉为 “21 世纪的石油”。然而&#xff0c;数据在推动社会进步的同时&#xff0c;也面临着前所未有的安全威胁。从 Facebook 超 5.33 亿用户数据泄露&#xff0c;到万…...

分享全国数字人才技能提升师资培训班 第五期邀请函

线下&#xff08;广州班&#xff09;&#xff1a; 大模型与AIGC多模态技术应用实战 线下&#xff08;青岛班&#xff09;&#xff1a; Deepseek教学应用与智能体开发实战 线上班&#xff08;十二大专题&#xff09;&#xff1a; DeepSeek大模型教学应用实战 大模型与AIGC技…...

Linux三剑客之grep命令使用教程

grep命令选项详解:从基础到进阶的实用指南 一、基本选项 1. -i:忽略大小写(Case Insensitive) 含义:搜索时不区分字母大小写。用法示例: 搜索包含"hello"的行,无论大小写:grep -i "hello" file.txt示例数据(file.txt):Hello World hello ther…...

Kotlin 极简小抄 P8(不可空类型、可空类型、注意事项、非空断言 !!)

Kotlin 概述 Kotlin 由 JetBrains 开发&#xff0c;是一种在 JVM&#xff08;Java 虚拟机&#xff09;上运行的静态类型编程语言 Kotlin 旨在提高开发者的编码效率和安全性&#xff0c;同时保持与 Java 的高度互操作性 Kotlin 是 Android 应用开发的首选语言&#xff0c;也可…...

【Spring AI集成实战】基于NVIDIA LLM API构建智能聊天应用:从配置到函数调用全解析

【Spring AI集成实战】基于NVIDIA LLM API构建智能聊天应用&#xff1a;从配置到函数调用全解析 前言 在人工智能应用开发领域&#xff0c;大语言模型&#xff08;LLM&#xff09;的集成能力至关重要。NVIDIA作为全球领先的GPU厂商&#xff0c;其LLM API提供了对Meta Llama-3.…...

git 删除某个远程库的分支

要删除 Git 远程仓库中的特定分支&#xff0c;可以通过以下步骤操作&#xff08;综合多个文档中的核心方法&#xff09;&#xff1a; ​1. 查看远程分支列表​ 首先确认目标分支是否存在&#xff1a; git branch -r # 显示所有远程分支&#xff08;格式为 origin/分支名&am…...

Redis实战-缓存篇(万字总结)

前言&#xff1a; 今天结合黑马点评这个项目&#xff0c;讲下有关Redis缓存的一些内容&#xff0c;例如缓存更新策略&#xff0c;缓存穿透&#xff0c;雪崩和击穿等。 今日所学&#xff1a; 什么是缓存缓存更新策略缓存穿透缓存雪崩缓存击穿缓存工具封存 目录 1.什么是缓存…...

QT5.15 MacOS 打包指南

QT5.15 MacOS 打包指南 在 MacOS 上打包 QT5.15 应用程序需要几个步骤&#xff0c;以下是详细说明&#xff1a; 1. 使用 macdeployqt 工具 QT 自带的 macdeployqt 工具可以自动处理大部分依赖关系&#xff1a; macdeployqt YourApp.app -dmg这会&#xff1a; 自动复制所需…...

Nginx location匹配模式详解

以下是对 Nginx location 匹配模式的详细说明及代码示例&#xff0c;包含注释解析&#xff1a; 1. 精确匹配&#xff08;Exact Match&#xff09; 语法: location /path { ... } 优先级: 最高&#xff0c;仅当请求路径与 /path 完全一致时触发。 location /login {# 仅匹配…...

Vue 3 路由传参使用指南

目录 一、路由传参概述 二、动态路由参数&#xff08;params&#xff09; 2.1 基础用法 2.2 传递参数 2.3 获取参数 2.4 可选参数 2.5 多个参数与正则约束 2.6 多 params 的详细用法 多个可选参数的使用 路由配置 获取可选参数 三、查询参数&#xff08;Query&#x…...