【设计模式】策略模式在Java工程中应用
创始人
2024-05-27 05:44:13
0

在之前的文章中,曾经给大家介绍过策略模式:【设计模式】策略模式,在该篇文章中,我们曾很清楚的说到,策略模式主要解决的问题是:在有多种算法相似的情况下,解决使用 if...else 所带来的复杂和难以维护。策略模式使我们的实现更加符合开闭原则:面向扩展开发,面向修改关闭。在后来的学习和工作的多次应用中,有了更加深刻的认识。再次总结此篇文章赘述下。

此篇文章,将以创建订单为例,创建不同类型的订单,不同类型的订单不同的处理逻辑。

类图:

代码:

  1. 抽象策略(Strategy)角色:抽象策略角色由抽象类或接口来承担,它给出具体策略角色需要实现的接口;

package strategyTest;import strategyTest.dto.BaseRequestDTO;/*** 类 名 称:CreateStrategy* 类 描 述:策略抽象类* 创建时间:2023/2/20 3:46 下午* 创 建 人:admin*/
public abstract class AbstractOrderStrategy {public abstract Boolean isPracticable(BaseRequestDTO dto);// 业务执行总流程public ResponseResult process(BaseRequestDTO dto) {ResponseResult paramsValidate = validateParams(dto);if (!paramsValidate.isSuccess()) {return paramsValidate;}ResponseResult ruleValidate = validateRules(dto);if (!ruleValidate.isSuccess()) {return ruleValidate;}ResponseResult processResult = doProcess(dto);if(processResult.isSuccess()){postProcessAction(dto);}return processResult;}// 参数校验protected abstract ResponseResult validateParams(BaseRequestDTO dto);// 业务规则校验protected abstract ResponseResult validateRules(BaseRequestDTO dto);// 核心实现protected abstract ResponseResult doProcess(BaseRequestDTO dto);// 后置处理(异步操作等)protected abstract void postProcessAction(BaseRequestDTO dto);
}

2.具体策略(ConcreteStrategy)角色:实现封装了具体的算法或行为;

顺风车类型处理:

package strategyTest.impl;import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Service;
import strategyTest.AbstractOrderStrategy;
import strategyTest.ResponseResult;
import strategyTest.dto.BaseRequestDTO;
import strategyTest.dto.SfcRequestDTO;/*** 类 名 称:SfcImpl* 类 描 述:TODO* 创建时间:2023/2/20 3:58 下午* 创 建 人:admin*/
@Service
public class SfcImpl extends AbstractOrderStrategy {private Logger logger = LoggerFactory.getLogger(SfcImpl.class);@Overridepublic Boolean isPracticable(BaseRequestDTO dto) {if(dto.getType() == 1){return true;}return false;}@Overrideprotected ResponseResult validateParams(BaseRequestDTO dto) {SfcRequestDTO sfcRequestDTO = (SfcRequestDTO) dto;if (sfcRequestDTO.getIsShare() == null || sfcRequestDTO.getIsShare() == 0) return ResponseResult.ERROR;logger.info("sfc validateParams success");return ResponseResult.SUCCESS;}@Overrideprotected ResponseResult validateRules(BaseRequestDTO dto) {// 无校验return ResponseResult.SUCCESS;}@Overrideprotected ResponseResult doProcess(BaseRequestDTO dto) {logger.info("sfc doProcess success");return ResponseResult.SUCCESS;}@Overrideprotected void postProcessAction(BaseRequestDTO dto) {logger.info("sfc 异步执行...");}}

巴士订单处理:

package strategyTest.impl;import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Service;
import strategyTest.AbstractOrderStrategy;
import strategyTest.ResponseResult;
import strategyTest.dto.BaseRequestDTO;
import strategyTest.dto.BusRequestDTO;/*** 类 名 称:BusImpl* 类 描 述:TODO* 创建时间:2023/2/20 3:58 下午* 创 建 人:admin*/
@Service
public class BusImpl extends AbstractOrderStrategy {private Logger logger = LoggerFactory.getLogger(BusImpl.class);@Overridepublic Boolean isPracticable(BaseRequestDTO dto) {if (dto.getType() == 2) {return true;}return false;}@Overrideprotected ResponseResult validateParams(BaseRequestDTO dto) {BusRequestDTO busRequestDTO = (BusRequestDTO) dto;if (busRequestDTO.getRouteId() == null || busRequestDTO.getRouteId() == 0) return ResponseResult.ERROR;logger.info("bus validateParams success");return ResponseResult.SUCCESS;}@Overrideprotected ResponseResult validateRules(BaseRequestDTO dto) {BusRequestDTO busRequestDTO = (BusRequestDTO) dto;if (busRequestDTO.getSeats() != null && busRequestDTO.getSeats() > 30) return ResponseResult.ERROR;logger.info("bus validateRules success");return ResponseResult.SUCCESS;}@Overrideprotected ResponseResult doProcess(BaseRequestDTO dto) {logger.info("bus doProcess success");return ResponseResult.SUCCESS;}@Overrideprotected void postProcessAction(BaseRequestDTO dto) {logger.info("bus 异步执行...");}}

3.上下文(Context)角色:持有抽象策略类的引用。(引用,执行策略)

package strategyTest;import org.springframework.beans.factory.BeanFactoryUtils;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationListener;
import org.springframework.context.event.ContextRefreshedEvent;
import org.springframework.stereotype.Component;
import strategyTest.dto.BaseRequestDTO;import java.util.ArrayList;
import java.util.List;
import java.util.Map;/*** 类 名 称:StrategyContext* 类 描 述:策略执行上下文* 创建时间:2023/2/20 3:47 下午* 创 建 人:admin*/
@Component
public class StrategyContext{// 策略管理类@Autowiredprivate List strategies = new ArrayList<>();// 获取策略类public AbstractOrderStrategy getStrategy(BaseRequestDTO dto) {for (AbstractOrderStrategy strategy : strategies) {if (strategy.isPracticable(dto)) {return strategy;}}return null;}}

调用方client:

可以在一个controller,也可以写在多个controller

package strategyTest.controller;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import strategyTest.AbstractOrderStrategy;
import strategyTest.ResponseResult;
import strategyTest.StrategyContext;
import strategyTest.dto.BaseRequestDTO;
import strategyTest.dto.BusRequestDTO;
import strategyTest.dto.SfcRequestDTO;/*** 类 名 称:客户端请:可以充当controller或者业务调用上游* 类 描 述:TODO* 创建时间:2023/2/20 3:58 下午* 创 建 人:admin*/
@RestController
public class ClientTest {@Autowiredprivate StrategyContext strategyContext;@RequestMapping("/createSfc")public ResponseResult createSfc(SfcRequestDTO sfcRequestDTO){if(sfcRequestDTO == null){return ResponseResult.ERROR;}return this.createOrder(sfcRequestDTO);}@RequestMapping("/createBus")public ResponseResult createBus(BusRequestDTO busRequestDTO){if(busRequestDTO == null){return ResponseResult.ERROR;}return this.createOrder(busRequestDTO);}private ResponseResult createOrder(BaseRequestDTO baseRequestDTO) {AbstractOrderStrategy strategy = strategyContext.getStrategy(baseRequestDTO);if(strategy == null){return ResponseResult.ERROR;}return strategy.process(baseRequestDTO);}
}

我们看下运行结果:

sfc运行结果:

bus运行结果:

是不是非常简洁明了!!!

我们看完代码,再结合类图看下,此时如果我们想新增一种类型的创建订单流程,比如创建一笔代驾订单,需要如何做? 我们只需要做两步:

1、新增一个代驾的实现类DjImpl,继承StrategyContext类;

2、在Controller中,新增一个方法(如果不同类型的订单有不在同一个controller,可将createOrder方法抽象到父类controller,此时新增一个子类controller继承父类和父类的方法即可);

其他地方无需改动。。。其他类型的创建订单流程完全无改动,无影响。下单流程解耦,测试范围极大的降低。

想想我们在项目中,相似的场景写过多少if else。每次添加新的实现,是否影响到原有功能。结合这篇文章,希望能对读者朋友有所帮助。

总结:

策略模式帮助我们把相似的算法抽象,使用上下文管理抽象类的具体实现,每次新增算法时,只需新增具体实现。使我们的代码轻松解耦,符合开闭原则,极大的缩小了影响范围。是成为一名优秀工程师的必修课!

相关内容

热门资讯

秘鲁总统任命前司法部长阿拉纳为... 秘鲁总统博鲁阿尔特(左)和新任总理爱德华多·阿拉纳(右)在宣誓仪式上新华社利马5月14日消息,秘鲁总...
原油:WTI下跌 美国原油库存...   原油下跌,此前有政府报告显示,美国原油库存创两个月来最大增幅,抵消了世界两大经济体贸易战暂停带来...
长江精工钢结构(集团)股份有限... 股票简称:精工钢构 股票代码:600496 编号:临2025-065转债简称:精工转债 转债代码...
山东高速路桥集团股份有限公司关... 证券代码:000498 证券简称:山东路桥 公告编号:2025-42山东高速路桥集团股份有限公司关...
丽珠医药集团股份有限公司关于回... 证券代码:000513、01513 证券简称:丽珠集团、丽珠医药 公告编号:2025-042丽珠医药...
因存在安全隐患,福特公司召回约... 央视记者当地时间5月14日获悉,福特汽车公司宣布,由于存在可能导致车辆在行驶过程中丧失制动功能、增加...
“土原料”变“潮产品” 青海兄弟乳业生物科技有限公司张成龙获“2025 IMW世界甜品巧克力&手工冰淇淋大赛暨2025意大利...
顺我者富 逆我者贫——解读特朗...   美国众议院共和党提出的新税收法案草案中,富人和企业投资者明显受益,而移民、顶尖私校等被特朗普长期...
青稞“链”上黄金道 产业升级加... 5月13日,海东市互助土族自治县青稞产业园内,青稞香飘满链。生产线轰鸣声中,金黄青稞粒经低温烘焙化作...
青海多措并举维护广告市场良好秩... 本报讯 (记者 董洁) 5月14日,记者从青海省市场监督管理局获悉,为切实维护全省广告市场秩序,保障...
厦门钨业股份有限公司关于向控股... 证券代码:600549 证券简称:厦门钨业 公告编号:临-2025-046厦门钨业股份有限公司关于...
安徽古麒绒材股份有限公司首次公... 安徽古麒绒材股份有限公司(以下简称“发行人”)首次公开发行不超过5,000万股人民币普通股(A股)(...
中蒙第二条跨境铁路开工 新华社呼和浩特5月14日电(记者 赵泽辉) 14日,中蒙两国能源合作重要通道——中蒙甘其毛都至嘎...
匠心与创新 本报记者 谭梅 刘珂瑜在青海兄弟乳业生物科技有限公司的生产车间里,看着张成龙专注调试冰淇淋配方的身影...
持续推动优质医疗资源扩容下沉 本报乌鲁木齐讯(全媒体记者 郑娅莉) 记者从自治区人民政府新闻办公室5月13日召开的新闻发布会上...
芜湖三联锻造股份有限公司关于投... 证券代码:001282 证券简称:三联锻造 公告编号:2025-031芜湖三联锻造股份有限公司关于...
国内首套氧化铝焙烧智能系统投运 原标题:国内首套氧化铝焙烧智能系统投运(记者滕继濮 实习记者夏天一 通讯员王晶)记者14日从中铝国际...
“头雁”竞飞强交流 经验互学促... “我常常在想,作为当代年轻人该如何实现自己的价值?到村任职的大半年时间,让我找到了答案——将个人理想...
《国务院2025年度立法工作计... 新华社北京5月14日电 经党中央、国务院同意,国务院办公厅日前印发《国务院2025年度立法工作计划...
我省举办《信访工作条例》集中宣... 本报讯 (记者 张晓英) 5月14日,省信访工作联席会议办公室、省信访局联合西宁市县两级信访部门在西...