spring boot策略模式实用: 告警模块为例

spring boot策略模式实用: 告警模块

0 涉及知识点

策略模式, 模板方法, 代理, 多态, 反射

1 需求概括

  • 场景: 每隔一段时间, 会获取设备运行数据, 如通过温湿度计获取到当前环境温湿度;
  • 需求: 对获取回来的进行分析, 超过配置的阈值需要产生对应的告警

2 方案设计

告警的类别往往容易变化, 比如今天只有温度过高告警, 明天可能就要增加一个温度过低告警, 所以设计最好可以满足开闭原则, 方便后续对功能进行删减;

整体思路如下:

  • 顶层接口Handler 定义了两个方法, check 用于校验是否应该产告警, 入参可以选择传入告警配置和需要判定对象, handle方法主要用于告警的具体处理过程, 如之前是否存在告警等;
  • 统一抽象类AbstractHandler 中, 重写handle 方法, 作为模版方法, 一般不同的告警处理流程是相近的, 可以抽象处理, 如都要判断进行判断当前是否已存在告警等; 抽象类中还可以抽象出通用的方法和声明通用属性;
  • 各个具体实现类, 如TemperatureHandler 等, 各类告警的具体实例对象, 如果告警判定方式或处理流程上有不同, 可以选择性的重写check 方法或handle方法, 由于java的多态, 程序运行时会选择正确处理方式;

为了保证模块的完整性, 增加代理类屏蔽告警的内部处理逻辑, 外部统一通过代理类调用;

3 代码实现

  • 接口

    java 复制代码
    /**
     * 顶级接口
     * @author lixiyuan
     */
    public interface Handler {
    
        boolean check(AlarmConfig config, Object data);
    
        void handle(AlarmConfig config, Integer id, Object data);
    }
  • 抽象类

    java 复制代码
    /**
     * 抽象类, 抽取通用字段/方法, 实现模板方法
     * @author lixiyuan
     */
    public class AbstractHandler implements Handler {
    
        @Autowired
        private CurrentAlarmService currentAlarmService;
    
        @Override
        public boolean check(AlarmConfig config, Object data) {
            return false;
        }
    
        @Override
        public void handle(AlarmConfig config, Integer id, Object data) {
            // 获取当前存在的告警
            CurrentAlarm current = currentAlarmService.getCurrentAlarmById();
            // 比较阈值
            boolean check = check(config, data);
    
            // 为true发生告警
            if (check) {
                if (current == null) {
                    // 创建告警
                    currentAlarmService.save();
                } else {
                    // 更新告警
                    currentAlarmService.update();
                }
            } else {
                if (current != null) {
                    // 结束告警
                    currentAlarmService.finish();
                }
            }
        }
    
        /**
         * 反射获取属性值
         */
        protected String getValueByField(Object obj, String fieldName) {
            if (obj == null) {
                return "";
            }
            try {
                Field field = obj.getClass().getDeclaredField(fieldName);
                field.setAccessible(true);
                Object value = field.get(obj);
                return value == null ? "" : value.toString();
            } catch (NoSuchFieldException | IllegalAccessException e) {
                System.out.println();
                throw new RuntimeException(e);
            }
        }
    
        /**
         * 如果判定方法比较通用, 也可以在这里定义好, 各个子类调用一下就行
         */
        protected boolean commonCompare() {
            return false;
        }
    }
  • 实现子类

    java 复制代码
    /**
     *
     * @author lixiyuan
     */
    @Component
    public class HumidityHandler extends AbstractHandler {
    
        @Override
        public boolean check(AlarmConfig config, Object data) {
            // 通过反射获取实时数据
            String humidity = getValueByField(data, "humidity");
            // 拿到配置中的阈值,然后比较, 略
            return false;
        }
    }
    
    
    /**
     *
     * @author lixiyuan
     */
    @Component
    public class TemperatureHandler extends AbstractHandler {
    
        @Override
        public boolean check(AlarmConfig config, Object data) {
            // 通过反射获取实时数据
            String humidity = getValueByField(data, "temperature");
            // 拿到配置中的阈值,然后比较, 略
            return false;
        }
    }

    代码详见: [完成代码](nanqiangli/wushixian (github.com))

相关推荐
neoooo9 分钟前
别慌,Java只有值传递——一次搞懂“为啥我改了它还不变”!
java·后端·spring
用户77853718369612 分钟前
一力破万法:从0实现一个http代理池
后端·爬虫
拖孩39 分钟前
微信群太多,管理麻烦?那试试接入AI助手吧~
前端·后端·微信
Humbunklung1 小时前
Rust枚举:让数据类型告别单调乏味
开发语言·后端·rust
radient1 小时前
Golang-GMP 万字洗髓经
后端·架构
蓝倾1 小时前
如何使用API接口实现淘宝商品上下架监控?
前端·后端·api
舂春儿1 小时前
如何快速统计项目代码行数
前端·后端
Pedantic1 小时前
我们什么时候应该使用协议继承?——Swift 协议继承的应用与思
前端·后端
Codebee1 小时前
如何利用OneCode注解驱动,快速训练一个私有的AI代码助手
前端·后端·面试
martinzh1 小时前
用Spring AI搭建本地RAG系统:让AI成为你的私人文档助手
后端