设计模式之策略模式-工作实战总结与实现

server/2024/11/27 4:10:44/

文章目录

  • 应用场景
  • 存在问题
  • 解决方案
  • 继续延伸

应用场景

假设有这样的业务场景,大数据系统把文件推送过来,根据不同类型采取不同的解析方式。多数的小伙伴就会写出以下的代码:

java">public class Question {public static void main(String[] args) {String type = "";if (type == "A") {//按照A格式解析} else if (type == "B") {//按B格式解析} else {//按照默认格式解析}}
}

存在问题

  • 1、如果分支变多,这里的代码就会变得臃肿,难以维护,可读性低。
  • 2、如果你需要接入一种新的解析类型,那只能在原有代码上修改。

说得专业一点的话,就是以上代码,违背了面向对象编程的开闭原则以及单一原则。

  • 开闭原则(对于扩展是开放的,但是对于修改是封闭的):增加或者删除某个逻辑,都需要修改到原来代码
  • 单一原则(规定一个类应该只有一个发生变化的原因):修改任何类型的分支逻辑代码,都需要改动当前类的代码。

解决方案

如果你的代码就是酱紫:有多个if…else等条件分支,并且每个条件分支,可以封装起来替换的,我们就可以使用策略模式来优化。

java">/*** 策略模式:* 一个接口或者抽象类,里面两个方法(一个方法匹配类型,一个可替换的逻辑实现方法)* 不同策略的差异化实现(就是说,不同策略的实现类)*/
public interface IFileStrategy {/*** 解析属于哪种文件类型,也可以继续优化** @return 文件类型*/String getFileType();/*** 封装的公用算法(具体的解析方法)** @param param*/void process(Object param);
}@Component
public class AFileResolve implements IFileStrategy {@Overridepublic String getFileType() {return DbDriverClassNameTypeEnum.MYSQL.getType();}@Overridepublic void process(Object param) {System.out.println("A类型文件解析具体逻辑");}
}@Component
public class BFileResolve implements IFileStrategy {@Overridepublic String getFileType() {return DbDriverClassNameTypeEnum.POSTGRES.getType();}@Overridepublic void process(Object param) {System.out.println("B类型文件解析具体逻辑");}
}@Component
public class DefaultResolve implements IFileStrategy {@Overridepublic String getFileType() {return DbDriverClassNameTypeEnum.ORACLE.getType();}@Overridepublic void process(Object param) {System.out.println("默认类型文件解析具体逻辑");}
}
java">/*** 如何使用呢?* 我们借助spring的生命周期,使用ApplicationContextAware接口,把对用的策略,初始化到map里面。然后对外提供resolveFile方法即可。* ApplicationContextAware接口能够轻松感知并在Spring中获取应用上下文,进而访问容器中的其他Bean和资源,* 这增强了组件间的解耦,了代码的灵活性和可扩展性,是Spring框架中实现高级功能的关键接口之一。*/
@Component
public class StrategyService implements ApplicationContextAware {private Map<String, IFileStrategy> iFileStrategyMap = new ConcurrentHashMap<>();/*** 初始化策略.** @param applicationContext Spring应用上下文* @throws BeansException*/@Overridepublic void setApplicationContext(ApplicationContext applicationContext) throws BeansException {Map<String, IFileStrategy> tmepMap = applicationContext.getBeansOfType(IFileStrategy.class);tmepMap.values().forEach(strategyService -> iFileStrategyMap.put(strategyService.getFileType(), strategyService));}/*** 统一对外提供的接口,根据类型处理文件** @param fileType* @param objectParam*/public void resolveFile(String fileType, Object objectParam) {IFileStrategy iFileStrategy = iFileStrategyMap.get(fileType);if (iFileStrategy != null) {iFileStrategy.process(objectParam);}}
}

继续延伸

在参数校验的场景中,经常会遇到根据某一个枚举字段的不同值来校验不同的入参字段,校验逻辑也是不同的,因此很容易通过IF ELSE来进行判断处理,但是这样的代码扩展性不强,CRUD痕迹很明显,不优雅!因此可以选择测罗模式进行优化。

java">public interface LinkageTriggerValidator {void check(LinkageRuleTriggerVo triggerVo);
}public class BizPeriodTypeValidator implements LinkageTriggerValidator {@Overridepublic void check(LinkageRuleTriggerVo triggerVo) {BizPeriodType eventType = triggerVo.getBizEventType();if (null == eventType) {throw new ParamCheckRuntimeException("业务周期事件配置缺少业务周期类型");}// todo}
}public class ReservationEventValidator implements LinkageTriggerValidator {@Overridepublic void check(LinkageRuleTriggerVo triggerVo) {ReservationEventType eventType = triggerVo.getReservationEventType();if (null == eventType) {throw new ParamCheckRuntimeException("预定事件配置缺少预定时间类型");}// todo}
}public abstract class LinkageTriggerValidatorContext {/*** 触发源校验器.*/private static Map<LinkageTriggerSource, LinkageTriggerValidator> triggerValidatorMap =new EnumMap<>(LinkageTriggerSource.class);static {triggerValidatorMap.put(LinkageTriggerSource.RESERVE_TIME, new ReservationEventValidator());triggerValidatorMap.put(LinkageTriggerSource.MANUAL_TRIGGER, new TriggerEventValidator());triggerValidatorMap.put(LinkageTriggerSource.BIZ_PERIOD, new BizPeriodTypeValidator());}public static void check(LinkageTriggerSource type, LinkageRuleTriggerVo triggerVo) {LinkageTriggerValidator validator = triggerValidatorMap.get(type);if (null == validator) {throw new UnsupportedOperationException("不支持该触发源类型:" + type);}validator.check(triggerVo);}
}
java">使用LinkageTriggerValidatorContext提供的check方法,统一进行参数校验即可
LinkageTriggerValidatorContext.check(trigger.getType(), trigger);

http://www.ppmy.cn/server/145233.html

相关文章

手机无法连接服务器1302什么意思?

你有没有遇到过手机无法连接服务器&#xff0c;屏幕上显示“1302”这样的错误代码&#xff1f;尤其是在急需使用手机进行工作或联系朋友时&#xff0c;突然出现的连接问题无疑会带来不少麻烦。那么&#xff0c;什么是1302错误&#xff0c;它又意味着什么呢&#xff1f; 1302错…

【Axure高保真原型】或和且条件

今天和大家分享或和且条件案例的原型模板&#xff0c;效果包括&#xff1a; 可以选择指标、等式和填写对应值构成条件等式&#xff1b; 点击添加条件按钮&#xff0c;可以增加一行新的条件&#xff1b; 点击所在行的号按钮&#xff0c;可以在该行下方添加一行新的条件&#x…

图像识别,摄像头选型,分辨率因素如何考量

在进行图像识别应用的开发过程中&#xff0c;选择合适的摄像头和分辨率是至关重要的。以下是一些考虑因素&#xff0c;可以帮助你做出更明智的选择&#xff1a; 1. 图像识别需求 对象大小&#xff1a;如果需要识别的小对象&#xff08;如蜗牛&#xff09;占比较小的像素面积&…

IDEA2024如何创建Web项目以及配置Tomcat

在Web项目的开发过程中&#xff0c;Tomcat作为一款开源的Servlet容器&#xff0c;扮演着至关重要的角色。它不仅能够提供稳定的运行环境&#xff0c;还支持多种Java EE规范&#xff0c;为开发者提供了丰富的功能支持。因此&#xff0c;正确配置Tomcat服务器对于确保Web项目的顺…

全面解读 Qt 容器类

全面解读 Qt 容器类&#xff1a;选择最合适的解决方案 Qt 提供了一组强大的容器类&#xff0c;用于存储和管理数据。这些容器类不仅功能强大&#xff0c;还特别针对 Qt 的需求进行了优化&#xff0c;在性能、使用便捷性和内存管理上具有独特优势。 在开发中&#xff0c;我们经…

ubuntu20.04中编译安装gcc 9.2.0

ubuntu20.04中编译安装gcc 9.2.0,步骤如下&#xff1a; #install compile dependence libraries 1&#xff1a;$ sudo apt install libgmp-dev libisl-dev libmpc-dev libmpfr-dev # install gcc 9.2.0 # download source code 2&#xff1a;$ wget http://ftp.gnu.org/gn…

鸿蒙系统ubuntu开发环境搭建

在RISC-V等平台移植鸿蒙系统OpenHarmony&#xff0c;需要使用linux环境进行代码的编译&#xff0c;为兼顾日常办公需要&#xff0c;可采用WindowsUbuntu虚拟机的混合开发的环境&#xff0c;通过网络及文件夹共享&#xff0c;在主机和虚拟机之间共享文件数据。 工具准备&#x…

Laravel对接SLS日志服务

Laravel对接SLS日志服务&#xff08;写入和读取&#xff09; 1、下载阿里云的sdk #通过composer下载 composer require alibabacloud/aliyun-log-php-sdk#对应的git仓库 https://github.com/aliyun/aliyun-log-php-sdk2、创建sdk请求的service <?phpnamespace App\Ser…