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

SpringBoot+策略模式+枚举类,使用配置文件改进,优雅消除if-else,完全符合OOP原则

需求分析

公司做物联网系统的,使用nettry进行设备连接,对设备进行数据采集,根据设备的协议对数据进行解析,解析完成之后存放数据库,但是不同厂家的设备协议不同。公司系统使用了使用了函数式编程的去写了一个解析类,所有的协议解析都在同一个方法类里面了,耦合度超高,不够优雅,我决定使用策略模式对它进行优化!

策略模式

下面是原理图,跟其他大佬的差不多,主要就是抽象策略接口具体策略环境三部分,这里不介绍策略模式了,建议去看看别的大佬写的好博客吧。
在这里插入图片描述

下面直接上代码!

代码实现

枚举类

现有的系统是根据协议长度来判断不同厂家的设备的,所以定义一个枚举类,len是协议的长度,beanName是具体策略的bean名称,通过传入的协议长度获取beanName,再获取具体策略的bean,去执行具体的解析方法。

public enum AnalysisEnum {

    SONGXIA(101, "songXiaAnalysisStrategy", "松下"),
    OTC(102, "OTCAnalysisStrategy", "OTC"),
    JIANGNAN(103, "jiangNanAnalysisStrategy", "江南"),

    ;


    private int len;
    private String beanName;
    private String desc;

    AnalysisEnum(int len, String beanName, String desc) {
        this.len = len;
        this.beanName = beanName;
        this.desc = desc;
    }

    public int getLen() {
        return len;
    }

    public String getBeanName() {
        return beanName;
    }

    public String getDesc() {
        return desc;
    }

    public static AnalysisEnum getAnalysisEnum(Integer len) {
        for (AnalysisEnum analysisEnum : AnalysisEnum.values()) {
            if (analysisEnum.getLen() == len) {
                return analysisEnum;
            }
        }

        throw new RuntimeException("异常");
    }
}

抽象策略接口和具体策略

public interface AnalysisStrategy {

    void analysis();
}

@Component("jiangNanAnalysisStrategy")
public class JiangNanAnalysisStrategy implements AnalysisStrategy {


    @Override
    public void analysis() {
        System.out.println("解析江南协议...");
    }
}

@Component("OTCAnalysisStrategy")
public class OTCAnalysisStrategy implements AnalysisStrategy {
    @Override
    public void analysis() {
        System.out.println("解析OTC协议...");
    }
}

@Component("songXiaAnalysisStrategy")
public class SongXiaAnalysisStrategy implements AnalysisStrategy {
    @Override
    public void analysis() {
        System.out.println("解析松下协议...");
    }
}

环境

抽象策略接口定义为Map<String, AnalysisStrategy>的value,通过spring的自动注入,所有的具体策略实现类都会被注入到map当中,key为beanId,即@Component指定的bean名称,配合AnalysisEnum,就可以通过协议长度获取具体策略执行具体方法,从而优雅地消除if-else

@Component
public class AnalysisContext {

    @Resource
    private Map<String, AnalysisStrategy> selectorMap;

    public void analysis(Integer len) {
        AnalysisEnum analysisEnum = AnalysisEnum.getAnalysisEnum(len);
        System.out.println("协议长度:" + len + " " + "设备:" + analysisEnum.getDesc());
        selectorMap.get(analysisEnum.getBeanName()).analysis();
    }
}

controller

@RestController
@Api(tags = "策略模式")
public class StrategyController {

    @Autowired
    private AnalysisContext analysisContext;

    @GetMapping("/strategy")
    @ApiOperation("策略模式测试接口")
    public void test1(@RequestParam Integer len) {
        analysisContext.analysis(len);
    }
}

测试结果

在这里插入图片描述

使用配置文件改进

可以看到,使用枚举类的话,想要扩展其他厂家的协议,就需要在枚举类中添加代码,违背了OOP原则。说实话,策略模式使用枚举类基本都是无一例外都是这样的问题。这个需求的难点主要在于协议长度与bean的映射,而扩展协议就需要扩展bean,就需要添加映射关系。如果把映射关系在配置文件中配置,扩展时再在配置文件中添加映射关系,不就可以了吗

application.yml 配置文件

analysis-strategy:
  len-bean-map:
    101: "songXiaAnalysisStrategy"
    102: "OTCAnalysisStrategy"
    103: "jiangNanAnalysisStrategy"

AnalysisStrategyConfig配置类

@Configuration
@ConfigurationProperties(prefix = "analysis-strategy")
@Data
public class AnalysisStrategyConfig {

    private Map<Integer, String> lenBeanMap;

    public Map<Integer, String> getLenBeanMap() {
        return lenBeanMap;
    }
}

修改之后的环境类

@Component
public class AnalysisContext {

    @Resource
    private AnalysisStrategyConfig analysisStrategyConfig;

    @Autowired
    private Map<String, AnalysisStrategy> strategyMap;

    public void analysis(Integer len) {
        String beanName = analysisStrategyConfig.getLenBeanMap().get(len);
        System.out.println("协议长度:" + len + " " + "设备:" + beanName);
        strategyMap.get(beanName).analysis();
    }
}

测试结果一模一样。
扩展只需要实践抽象策略接口,在配置文件中添加映射关系即可,无需球盖源代码。
优雅,实在是太优雅了!
在这里插入图片描述

总结

最优雅地策略模式,当然是实现了抽象策略接口之后就可以使用,但是目前本帅写的代码当中尽管优雅地消除了if-else,但不符合OOP原则,不过没办法,系统中就必须得通过协议长度去判断不同的协议,而spring中注入具体策略类是通过beanId去实现的,无法通过协议长度直接拿到具体策略类。有一个好的方法就是通过配置文件或者数据库去配置协议长度和beanId的关系,这样子以后再有扩展不同厂家的协议之后就可以不需求改动源代码,而只需要在配置文件或数据库中添加协议长度和beanId的映射关系即可

相关文章:

  • nVisual对接企业微信实现机房设备与连接变更的自动化审批
  • 计算机网络--第四章 网络层(2)
  • Spring-Mybatis框架常见面试题
  • 华为云 对象存储服务 OBS | 架构分析与应用场景
  • 夯实 kafka 系列|第二章:kafka 常用参数配置
  • 【leetcode hot 100 33】搜索旋转排序数组
  • 当汉堡遇上便当:TypeScript命名空间 vs JavaScript模块化
  • 机器学习-基于KNN算法手动实现kd树
  • qt中libusb热插拔检测示例代码
  • 【机器学习】什么是随机森林?
  • Linux第零节:Linux命令速查图表(按功能分类)
  • go的参数传递都是值传递,但切片需要注意
  • C++ 性能优化隐藏危机:忽视数据结构与内存细节,效率大打折扣
  • 【前端】在<el-form>里循环插入list内容
  • 百度富文本编辑器配置(vue3)
  • Ubuntu20.04 Qt5安装和卸载
  • uniapp处理流式请求
  • 【计算机网络】-计算机网络期末复习题复习资料
  • WebRTC协议全面教程:原理、应用与优化指南
  • 专访海鹏科技董事长秘书、服务总监赵静波:前瞻式智能化管理,为全球售后服务保驾护航
  • “敌人已经够多了”,菲总统马科斯:愿与杜特尔特家族和解
  • 陈龙带你观察上海生物多样性,纪录片《我的城市邻居》明播出
  • 证监会副主席李明:近期将出台深化科创板、创业板改革政策措施
  • 俄外长与美国务卿通电话,讨论俄美接触等问题
  • 著名文博专家吴远明因交通事故离世,享年75岁
  • 爬坡难下坡险,居民出行难题如何解?