行为型设计模式-策略模式(Strategy Pattern)

策略模式

策略模式:百度百科中引述为:指对象有某个行为,但是在不同的场景中,该行为有不同的实现算法。

  1. 策略模式是对算法的包装,是把使用算法的责任和算法本身分割开来,委派给不同的对象管理。
  2. 策略模式通常把一个系列的算法包装到一系列的策略类里面,作为一个抽象策略类的子类。用一句话来说,就是:“准备一组算法,并将每一个算法封装起来,使得它们可以互换”。【此处的算法,可以理解为解决业务需求的方法。】

 换一种说法就是:一个类的行为或其算法可以在运行时更改。我们把它降维到代码层面, 用人话翻译一下就是 ,运行时我给你这个类的方法传不同的“key”,你这个方法会执行不同的业务逻辑。细品一下,这不就是 if else 干的事吗?

 举个实际的例子:审核流程,请假和调休都是提交审核>审批,这个审批的时候要干的事就不同了。如果你传的Type是请假,诶!那就要扣你工资了,起码全勤是没了。如果type是调休的话那就没事了,工资照常发。那正常的代码结构一般就是下面这样了

if(请假){//todo 扣你工资xxx
} else if(调休){//todo 工资照发
}

其实策略模式的核心思想和 if else如出一辙,根据不同的key动态的找到不同的业务逻辑,那它就只是如此吗

实际上,我们口中的策略模式其实就是在代码结构上调整,用接口+实现类+分派逻辑来使代码结构可维护性好点。

在这里插入图片描述

一般教科书上讲解到接口与实现类就结束了,其他博客上会带上提及分派逻辑。这里就不啰嗦了。

小结一下,即使用了策略模式,你该写的业务逻辑照常写,到逻辑分派的时候,还是变相的if else。而它的优化点是抽象了出了接口,将业务逻辑封装成一个一个的实现类,任意地替换。在复杂场景(业务逻辑较多)时比直接 if else 来的好维护些。

就是几个if else场景我需要用到策略模式?

 我想小伙伴们经常有这样的不满,我的业务逻辑就3 4 行,你给我整一大堆类定义?有必要这么麻烦吗?我看具体的业务逻辑还需要去不同的类中,简单点行不行。

其实我们所不满的就是策略模式带来的缺点:

1、策略类会增多
2、业务逻辑分散到各个实现类中,而且没有一个地方可以俯视整个业务逻辑

 针对传统策略模式的缺点,在这分享一个实现思路,这个思路已经帮我们团队解决了多个复杂if else的业务场景,理解上比较容易,代码上需要用到Java8的特性——利用Map与函数式接口来实现。

直接show代码结构:为了简单演示一个思路,代码用String 类型来模拟一个业务BO

import org.springframework.stereotype.Service;import javax.annotation.PostConstruct;
import java.util.HashMap;
import java.util.Map;
import java.util.function.Function;@Service
public class TestService {/*** 业务逻辑分派Map* Function为函数式接口,下面代码中 Function<String, String> 的含义是接收一个Stirng类型的变量,返回一个String类型的结果*/private Map<String, Function<String, String>> checkResultDispatcher = new HashMap<>();/*** 初始化 业务逻辑分派Map 其中value 存放的是 lambda表达式*/@PostConstructpublic void checkResultDispatcherInit() {checkResultDispatcher.put("请假", type -> String.format("%s扣你工资", type));checkResultDispatcher.put("调休", type -> String.format("%s不扣你工资", type));}public String getCheckResultSuper(String type) {//从逻辑分派Dispatcher中获得业务逻辑代码,result变量是一段lambda表达式Function<String, String> result = checkResultDispatcher.get(type);if (result != null) {//执行这段表达式获得String类型的结果return result.apply(type);}return "不正确的业务类型";}}

测试:

@RestController
public class TestCtrl {@AutowiredTestService testService;@PostMapping("/v1/demo/test")public String test2(String type) {return testService.getCheckResultSuper(type);}
}

使用接口测试工具得到如下结果:

在这里插入图片描述

鲁迅曾说过,“每解决一个问题,就会引出更多的问题”。我们一起来看看这样的实现有什么好处,会带来什么问题。

在这里插入图片描述

好处很直观

  1. 在一段代码里直观的看到"判断条件"与业务逻辑的映射关系
  2. 不需要单独定义接口与实现类,直接使用现有的函数式接口(什么?不知道函数式接口?快去了解),而实现类直接就是业务代码本身。

不好的点

  1. 需要团队成员对lambda表达式有所了解(什么?Java17都出来了还有没用上Java8新特性的小伙伴?)

接下来我举几个在业务中经常遇到的if else场景,并用Map+函数式接口的方式来解决它
有的小伙伴会说,我的判断条件有多个啊,而且很复杂,你之前举个例子只有单个判断逻辑,而我有多个判断逻辑该怎么办呢?
很好解决:写一个判断逻辑的方法,Map的key由方法计算出

@Service
public class TestService {/*** 业务逻辑分派Map* Function为函数式接口,下面代码中 Function<String, String> 的含义是接收一个Stirng类型的变量,返回一个String类型的结果*/private Map<String, Function<String, String>> checkResultDispatcher = new HashMap<>();private static String  QJ_PASS = "请假_通过";private static String  QJ_REJECT = "请假_驳回";private static String  TX_PASS = "调休_通过";private static String  TX_REJECT = "调休_驳回";/*** 初始化 业务逻辑分派Map 其中value 存放的是 lambda表达式*/@PostConstructpublic void checkResultDispatcherInit() {checkResultDispatcher.put(QJ_PASS, type -> String.format("%s成功,扣你工资", type));checkResultDispatcher.put(QJ_REJECT, type -> String.format("%s失败,老实上班", type));checkResultDispatcher.put(TX_PASS, type -> String.format("%s成功,不扣你工资,放心去浪", type));checkResultDispatcher.put(TX_REJECT, type -> String.format("%s失败,老实上班", type));}public String getCheckResultSuper(String type, String state) {//从逻辑分派Dispatcher中获得业务逻辑代码,result变量是一段lambda表达式String key = getDispatcherKey(type,state);Function<String, String> result = checkResultDispatcher.get(key);if (result != null) {//执行这段表达式获得String类型的结果return result.apply(type);}return "不正确的业务类型";}/*** 判断条件方法*/private String getDispatcherKey(String type, String state) {return type + "_" + state;}}

测试 controller修改如下:

@PostMapping("/v1/demo/test")
public String test2(String type, String state) {return testService.getCheckResultSuper(type,state);
}

测试结果如下所示:

在这里插入图片描述

可以看出,只要设计好key的生成规则,多判断逻辑的需求是完全可以满足的。
既然鲁迅说过,“每解决一个问题,就会引出更多的问题”。那么我们接下来看看还有什么问题

如果我的业务逻辑有很多很多行,在checkResultDispatcherMuitInit()方法的Map中直接写不会很长吗?
直接写当然长了,我们可以抽象出一个service服务专门放业务逻辑,然后在定义中调用它就好了:

@Service
public class BizUnitService {public String qjPass(String type) {return type + "通过+各种花式操作";}public String qjReject(String type) {return type + "失败+各种花式操作";}public String txPass(String type) {return type + "成功+各种花式操作";}public String txReject(String type) {return type + "失败+各种花式操作";}}
@Service
public class TestService {@AutowiredBizUnitService bizUnitService;/*** 业务逻辑分派Map* Function为函数式接口,下面代码中 Function<String, String> 的含义是接收一个Stirng类型的变量,返回一个String类型的结果*/private Map<String, Function<String, String>> checkResultDispatcher = new HashMap<>();private static String  QJ_PASS = "请假_通过";private static String  QJ_REJECT = "请假_驳回";private static String  TX_PASS = "调休_通过";private static String  TX_REJECT = "调休_驳回";/*** 初始化 业务逻辑分派Map 其中value 存放的是 lambda表达式*/@PostConstructpublic void checkResultDispatcherInit() {checkResultDispatcher.put(QJ_PASS, type -> bizUnitService.qjPass(type));checkResultDispatcher.put(QJ_REJECT, type -> bizUnitService.qjReject(type));checkResultDispatcher.put(TX_PASS, type -> bizUnitService.txPass(type));checkResultDispatcher.put(TX_REJECT, type -> bizUnitService.txReject(type));}public String getCheckResultSuper(String type, String state) {//从逻辑分派Dispatcher中获得业务逻辑代码,result变量是一段lambda表达式String key = getDispatcherKey(type,state);Function<String, String> result = checkResultDispatcher.get(key);if (result != null) {//执行这段表达式获得String类型的结果return result.apply(type);}return "不正确的业务类型";}/*** 判断条件方法*/private String getDispatcherKey(String type, String state) {return type + "_" + state;}}

测试结果如下所示:
在这里插入图片描述

道不行,乘桴浮于海.    --《论语·公冶长》

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/285443.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

AttributeError: ‘module‘ object has no attribute ‘RAW_OPT‘解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

定制美化ThingsBoard中的部件库中的Card,HTML,Markdwon部件

thingsbaord中的部件库,有些默认样式并不美观,这时候就需要我们编写一部分CSS进行美化。看一下默认的样式 简单美化后的样式 修改部件配置,在部件的css或者部件的style上 编写css。不光应用于属性卡片部件,其他部件同样适用。 .widget-type-sys-cards-attributes_car…

如何在Eclipse中安装WindowBuilder插件,详解过程

第一步&#xff1a;找到自己安装eclipse的版本&#xff0c;在Help-关于eclipse里面&#xff0c;即Version 第二步&#xff1a;去下面这个网站找到对应的 link&#xff08;Update Site&#xff09;&#xff0c;这一步很重要&#xff0c;不然版本下载错了之后还得删除WindowBuil…

数据结构--图

树具有灵活性&#xff0c;并且存在许多不同的树的应用&#xff0c;但是就树本身而言有一定的局限性&#xff0c;树只能表示层次关系&#xff0c;比如父子关系。而其他的比如兄弟关系只能够间接表示。 推广--- 图 图形结构中&#xff0c;数据元素之间的关系是任意的。 一、图…

10 v-html指令

概述 v-html主要是用来渲染富文本内容&#xff0c;比如评论信息&#xff0c;新闻信息&#xff0c;文章信息等。 v-html是一个特别不安全的指令&#xff0c;因为它会将文本以HTML的显示进行渲染&#xff0c;一旦文本里面包含一些恶意的js代码&#xff0c;可能会导致整个网页发…

YOLOv8加入顶会ICLR2022MobileViT模块

一、原文引入介绍 MOBILEVIT:轻量级、通用型且移动友好的视觉Transformer 论文地址:https://arxiv.org/pdf/2110.02178.pdf MOBILEVIT: LIGHT-WEIGHT, GENERAL-PURPOSE,AND MOBILE-FRIENDLY VISION TRANSFORMER MobileViT是由苹果公司发表在ICLR2022顶会上的一篇文章,这篇文…

机器学习:增强式学习Reinforcement learning

收集有标签数据比较困难的时候同时也不知道什么答案是比较好的时候可以考虑使用强化学习通过互动&#xff0c;机器可以自己知道什么结果是好的&#xff0c;什么结果是坏的 Outline 什么是RL Action就是一个functionEnvironment就是告诉这个Action是好的还是坏的 例子 Space i…

竞赛保研 python+opencv+深度学习实现二维码识别

0 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 &#x1f6a9; pythonopencv深度学习实现二维码识别 &#x1f947;学长这里给一个题目综合评分(每项满分5分) 难度系数&#xff1a;3分工作量&#xff1a;3分创新点&#xff1a;3分 该项目较为新颖&…

mysql自动安装脚本(快速部署mysql)

mysql_install - 适用于生产环境单实例快速部署 MySQL8.0 自动安装脚本 mysql8_install.sh&#xff08;执行前修改一下脚本里的配置参数&#xff0c;改成你自己的&#xff09;&#xff08;博客末尾&#xff09; my_test.cnf&#xff08;博客末尾&#xff09;&#xff08;这个…

工作:三菱PLC程序开发流程总结

工作&#xff1a;三菱PLC程序开发流程总结 一、程序流程图 程序流程图是逻辑思维与动作流程的检查图&#xff0c;是保证逻辑思维合理的前提&#xff0c;写代码丢失方向可从程序流程图重新整理&#xff0c;程序流程图非常重要。 二、组态配置 组态配置是将所用到的基板和模块…

基于Java SpringBoot和Vue的医院信息管理挂号系统

摘要 医院信息管理系统&#xff08;Hospital Information Management System&#xff0c;简称HIMS&#xff09;是一种应用于医疗机构的信息化管理系统&#xff0c;旨在提高医疗服务质量、降低运营成本、提高工作效率和满足患者需求。HIMS通过对医院内各种信息的集成、管理和共享…

C++11 【初识】

C11简介 1.在2003年C标准委员会曾经提交了一份技术勘误表(简称TC1)&#xff0c;使得C03这个名字已经取代了C98称为C11之前的最新C标准名称。 2.不过由于C03(TC1)主要是对C98标准中的漏洞进行修复&#xff0c;语言的核心部分则没有改动&#xff0c;因此人们习惯性的把两个标准合…