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

行为型模式(一)策略模式

目录

1. 介绍

2. 目的

3. 实践


1. 介绍

策略模式(Strategy Pattern)是一种行为设计模式,它定义了一系列算法,将每种算法封装成具体的策略类,并使它们可以互相替换。策略模式使得算法可以独立于客户端而变化,客户端可以在运行时动态地选择不同的策略来执行相应的算法。

在策略模式中,通常包括以下几个角色:

  1. Context(上下文): 上下文类持有一个策略对象的引用,它将客户端请求委派给具体的策略对象来执行相应的算法。上下文类通常包含一个可以动态切换策略的方法,以便在运行时选择不同的策略。
  2. Strategy(策略): 策略接口或抽象类定义了一个算法族,其中每个具体的策略类实现了这个接口或继承了这个抽象类。策略类封装了具体的算法实现,客户端通过上下文类调用策略对象的方法来执行算法。

  3. ConcreteStrategy(具体策略): 具体策略类实现了策略接口或抽象类,它包含了具体的算法实现。每个具体策略类实现了一种特定的算法,可以根据具体需求选择不同的具体策略类来执行相应的算法。

 策略模式的核心思想是将算法的实现和使用分离,使得算法可以独立变化而不影响客户端代码。通过策略模式,可以更好地组织和管理复杂的算法逻辑,提高代码的灵活性、可维护性和可扩展性。策略模式常用于需要动态地切换算法或根据不同条件选择不同算法的场景。

2. 目的

  • 消除大量的条件判断语句: 当代码中存在大量的if-else语句用于根据不同条件执行不同的逻辑时,策略模式可以帮助将不同的条件逻辑封装在不同的策略类中,从而避免代码臃肿和难以维护。
  • 提高代码的灵活性和可扩展性: 策略模式使得具体的策略类可以独立变化,客户端可以在运行时动态地选择不同的策略对象,从而实现系统的灵活性和可扩展性。

  • 避免代码重复: 将公共部分抽象到策略接口中,具体策略类只需实现自己特有的部分。

  • 提高代码的可读性和可维护性: 策略模式将算法的实现细节隐藏在具体策略类中,使得代码结构更清晰,易于理解和维护。

总的来说,策略模式可以帮助解决多个条件判断导致的代码复杂性、提高代码的灵活性和可扩展性,以及避免代码重复,从而提高代码质量和可维护性。

3. 实践

医嘱打印功能,多个打印模板,后续还会增加模板。

针对不同打印模板,进行不同数据处理。

使用策略模式就能很好解决此类问题。

//策略接口
public interface IPrintMethod {List<BiOrderPrint> doPrint(BiOrderPrintSaveBO bo, SysDicItemDTO dicItemDTO);default Map<String, List<BiOrderPrint>> toMap(List<BiOrderPrint> list) {return list.stream().collect(Collectors.groupingBy(BiOrderPrint::getFyFlag));}default List<BiOrderPrint> toList(Map<String, List<BiOrderPrint>> map) {return map.values().stream().flatMap(List::stream).collect(Collectors.toList());}
}//策略实现类1
@Component
@RequiredArgsConstructor
public class PqMethod implements IPrintMethod {private final HandlerOne handlerOne;private final HandlerTwo handlerTwo;private final HandlerThree handlerThree;private final HandlerFour handlerFour;private final HandlerFive handlerFive;private final HandlerLast handlerLast;@Overridepublic List<BiOrderPrint> doPrint(BiOrderPrintSaveBO bo, SysDicItemDTO dicItemDTO) {//1. 数据转换,数据预处理List<BiOrderPrint> biOrderPrintList = handlerOne.doHandler(bo);Map<String, List<BiOrderPrint>> map1 = toMap(biOrderPrintList);//2. 数据填充Map<String, List<BiOrderPrint>> map2 = handlerTwo.doHandler(map1);//3. 根据每页条数,分割部分Map<String, List<BiOrderPrint>> map3 = handlerThree.doHandler(map2, dicItemDTO);//4. 每个分页里面的数据均是小于等于n的,将数据补 全Map<String, List<BiOrderPrint>> map4 = handlerFour.doHandler(map3, dicItemDTO);List<BiOrderPrint> resultList = toList(map4);// 5. 处理特殊数据handlerFive.doHandler(bo);// 6. 报存return handlerLast.doHandler(resultList);}
}//策略实现类2
@Component
@RequiredArgsConstructor
public class SyMethod implements IPrintMethod {private final HandlerOne handlerOne;private final HandlerTwo handlerTwo;private final HandlerLast handlerLast;@Overridepublic List<BiOrderPrint> doPrint(BiOrderPrintSaveBO bo, SysDicItemDTO dicItemDTO) {// 1. 数据转换List<BiOrderPrint> biOrderPrintList = handlerOne.doHandler(bo);// 2. 过滤相同orderId的数据,只保留一份List<BiOrderPrint> list1 = new ArrayList<>(biOrderPrintList.stream().collect(Collectors.toMap(BiOrderPrint::getOrderId,Function.identity(),(existing, replacement) -> existing)).values());// 3. 数据填充Map<String, List<BiOrderPrint>> map1 = toMap(list1);Map<String, List<BiOrderPrint>> map2 = handlerTwo.doHandler(map1);List<BiOrderPrint> resultList = toList(map2);// 4. 报存return handlerLast.doHandler(resultList);}
}//......
//Context(上下文)
@Service
@RequiredArgsConstructor
public class OrderPrintContext {private final List<IPrintMethod> IPrintMethods;public List<BiOrderPrint> doPrint(BiOrderPrintSaveBO bo, SysDicItemDTO dicItemDTO) {//将所有策略实现类放到map里,前端传具体策略Map<String, IPrintMethod> printMethodMap = IPrintMethods.stream().collect(Collectors.toMap(IPrintMethod -> IPrintMethod.getClass().getSimpleName(), Function.identity()));if (ObjectUtil.isNull(dicItemDTO)) {throw new ServiceException(OrderErrorEnum.ERROR_82003, bo.getBillType());}IPrintMethod IPrintMethod = printMethodMap.get(dicItemDTO.getValue3());return IPrintMethod.doPrint(bo, dicItemDTO);}
}

http://www.lryc.cn/news/416994.html

相关文章:

  • JAVA中的Wrapper类
  • 在没有硬盘的情况下进行电脑数据迁移
  • C++转Java基础知识
  • 搭建jenkins一键部署java项目
  • 从零到一打造自己的大模型(一)模型实现
  • 【开源项目】基于RTP协议的H264码流发送器和接收器
  • 【C++】4.类和对象(2)
  • 搭建基于树莓派的Linux学习环境(TODO)
  • 《大电机技术》是什么级别的期刊?是正规期刊吗?能评职称吗?
  • Python 中使用 Split 忽略逗号
  • YOLOv10改进 | 主干篇 | YOLOv10引入CVPR2023 顶会论文BiFormer用于主干修改
  • sql注入靶场搭建
  • 【MySQL】MySQL的JSON特性
  • 微信小程序 - 自定义计数器 - 优化(键盘输入校验)
  • Nacos 容器化安装和代理配置指南
  • css水波浪动画效果
  • SQL二次注入
  • 深入学习小程序开发第二天:数据绑定与动态更新
  • 【ai】 时间序列分析的python例子
  • 生成订单幂等性(防止订单重复提交)
  • IDEA自定义注释模版
  • Spring Cloud Gateway实现API访问频率限制
  • 单例模式:确保唯一实例的设计模式
  • MCU调试技巧-串口打印
  • VS+Qt+C++点云PCL三维显示编辑系统
  • 代码随想录算法训练营第七天(一)| 454.四数相加II 383. 赎金信
  • SpringBoot+Mybatis 分页
  • 学习进行到了第十七天(2024.8.5)
  • 【Nuxt】Layout 布局和渲染模式
  • C:指针学习(1)-学习笔记