JavaWeb开发经验谈:业务行为相似DAO接口的统一封装与使用

简介: 以对mybatis-plus的封装使用为例,给出一个在适用业务场景下大幅降低代码信息熵的解决方案

一个典型的web应用中可能会有很多DAO接口(也被称作mapper,以下皆称mapper),在某些业务场景中,我们会对不同的mapper进行高度相似的业务操作,这种情况下如果仍然裸用mapper进行CRUD,就可能制造出大批高信息熵的代码,久而久之,难以维护.

例如在笔者之前所开发的项目,它是一个云产品监控系统,通过组合阿里云的blink、datahub、TableStore以及一些java微服务,实现了对几十种阿里云产品的数百个监控项的统一配置实时监控和按规则报警.

为了便于blink服务分监控项计算受监控实例的某项指标是否突破阈值,我们存储实例阈值配置的表结构被设计以实例ID作为唯一标识,横向扩展监控项的横表结构:

实例ID

监控项1

监控项2

监控项3

更多监控项...

实例1

55

66

77

88

实例2

更多实例...

并且,让每一种产品都对应一张阈值横表,便于维护,也避免了给不同产品的同名监控项起别名(比方说,ECS和RDS都有内存使用率这个监控项)维护的成本.

在这个场景中,负责报警配置的业务开发人员事实上并不需要关心每个监控项的具体业务含义,他可以对监控项作一个统一的抽象:

// 由于历史原因,我们的项目为每一个监控项都创建了对应的VO,// 这样很繁琐,但是对于分类维护监控项或在domain层次针对性地扩展某个监控项则是有利的// 如果你偏好贫血模型,那么可以尝试用一个CommonMonitorConfig描述所有publicabstractclassCommonMonitorConfig {
// 超阈值状态超过此时间,则进行报警StringkeepTime;
// 自上一次发送报警后的沉默周期StringsilenceTime;
// 此监控项是否启用报警BooleanisOpen;
// 监控项对应的字段名称StringmonitorName;
//  配置阈值Doublevalue;
//  产品类型,贫血模型需要此字段StringproductType;
//  充血模型可以定义此方法,让处于产品及的监控类例如EcsMonitorConfig实现此方法    abstractTypeEnumgetTypeEnum();
publicCommonMonitorConfig(){
// 如果使用充血模型,你可以分门别类的在构造方法中按类型初始化一些默认配置    }
}

前端可以根据此抽象制作操作报警配置的表单界面并提交报警配置数据.在约束好CommonMonitorConfig的monitorName与阈值配置实例类相应字段的映射关系后,后端接口中可以使用反射方便规整地将VO对象的相应值映射统一到对应的DO对象中.

当然,在真正实现丝滑流畅无污染的反射之前还需要作一点点合理封装.

以实例阈值配置的存储和查询为例,首先给所有产品的entity类定义一个公共父类,并提取公共业务字段到父类中,使得子类实体只存放阈值:

publicclassCommonInsAlarmRule {
@TableIdpublicLongid;
/*** 规则编码*/publicStringruleUUID;
/*** 实例ID*/publicStringinstanceId;
publicStringmoduleCode;
}

然后,写一个ConfigUtil维护一些静态映射关系并封装常用方法:

/*** k: 产品类别* v: 报警规则实体类*/publicstaticfinalMap<TypeEnum, Class<?extendsCommonInsAlarmRule>>insClassMap=newImmutableMap.Builder<CloudProductTypeEnum, Class<?extendsCommonInsAlarmRule>>()
            .put(ECS, EcsAlarmRule.class)
            .put(EIP, EipAlarmRule.class)
            .put(RDS, RdsAlarmRule.class)
// ....            .build();
// 比方说,从上面的字典中获取insType,然后将其实例化为被标记为CommonInsAlarmRule的具体子类对象publicstatic<P, SextendsP>PnewInstance(Class<S>insType) {
try {
returninsType.newInstance();
        } catch (InstantiationException|IllegalAccessExceptione) {
        }
    }
// 这个方法直接定义在TypeEnum这个枚举类里publicstaticTypeEnumfindEnumByNameIgnoreCase(StringdisplayName){
returnArrays.stream(TypeEnum .values()).filter(e->e.displayName.equalsIgnoreCase(displayName)).findFirst().orElse(null);
    }

同样地,在ConfigUtil中注入并初始话各个产品mapper对象(这里以MybatisPlus为例)的映射关系:

privatestaticMap<String, BaseMapper<?extendsCommonInsAlarmRule>>instanceAlarmRuleMapperMap;
privatefinalEcsAlarmRuleMapperecsAlarmRuleMapper;
privatefinalRdsAlarmRuleMapperrdsAlarmRuleMapper;
privatefinalEipAlarmRuleMappereipAlarmRuleMapper;
//.... @PostConstructvoidinit(){
instanceAlarmRuleMapperMap=newImmutableMap.Builder<TypeEnum, BaseMapper<?extendsCommonInsAlarmRule>>()
                .put(EcsAlarmRule.class.getName(), ecsAlarmRuleMapper)
                .put(EipAlarmRule.class.getName(), eipAlarmRuleMapper)
                .put(RdsAlarmRule.class.getName(), rdsAlarmRuleMapper)
// .....                .build();
}

为了更好地利用MybatisPlus的LambdaQueryWrapper静态字段缓存进行代码中相关SQL的静态检查,需要为CommonInsAlarmRule创建一个无需对应任何数据库表的形式化mapper:

importcom.baomidou.mybatisplus.core.mapper.BaseMapper;
importorg.springframework.stereotype.Repository;
@RepositorypublicinterfaceAbstractInsMapperextendsBaseMapper<CommonInsAlarmRule> {
}


最后,在ConfigUtil中封装增删查改的常用方法:

publicstatic<TextendsCommonInsAlarmRule>BaseMapper<T>getInsAlarmRuleMapperByInsType(TypeEnumproductType) {
Class<T>clazz= (Class<T>) insClassMap.get(productType);
return (BaseMapper<T>) instanceAlarmRuleMapperMap.get(clazz.getName());
    }
publicstaticList<CommonInsAlarmRule>selectList(CloudProductTypeEnumproductType, LambdaQueryWrapper<CommonInsAlarmRule>wrapper) {
returngetInsAlarmRuleMapperByInsType(productType).selectList(wrapper);
    }
publicstaticCommonInsAlarmRuleselectOne(CloudProductTypeEnumproductType, LambdaQueryWrapper<CommonInsAlarmRule>wrapper) {
returngetInsAlarmRuleMapperByInsType(productType).selectOne(wrapper);
    }
publicintinsertOne(CloudProductTypeEnumproductType, CommonInsAlarmRulerule){
returngetInsAlarmRuleMapperByInsType(productType).insert(rule);
    }

举个例子,下图是一段根据实例ID及产品类型查询指定实例的配置阈值的代码,可以明显看出,这段代码的信息熵会随着产品数量的增加而逐渐升高:

1.png

使用统一封装之后的接口来替代原先的这段代码,只需三行,且可在后续迭代中保持信息密度不变:

2.png

相关文章
|
10天前
|
安全 Java 编译器
Java的封装详解
封装和多态是面向对象编程(OOP)的重要概念。封装通过私有属性和公共方法实现数据隐藏和保护,使类的内部细节对外部不可见;多态则通过方法重载和重写实现同一方法在不同对象上的不同表现形式,增强了代码的灵活性和可维护性。两者结合使用,可以使Java程序更加安全、灵活且易于维护。
|
10天前
|
Java
Java的封装详解
封装是Java中实现数据隐藏和保护的核心机制。它通过将对象的状态和行为结合并限制外部直接访问,确保类的内部细节对外不可见,仅能通过公共方法访问和修改对象状态。封装带来了数据隐藏、提高代码可维护性和增强安全性等好处。在Java中,封装主要通过将属性设为私有并提供getter和setter方法来实现。这种方式不仅保护了数据完整性,还允许在修改类内部实现时不影响外部代码,从而提升程序的健壮性和可读性。
|
7天前
|
设计模式 Java 关系型数据库
【Java笔记+踩坑汇总】Java基础+JavaWeb+SSM+SpringBoot+SpringCloud+瑞吉外卖/谷粒商城/学成在线+设计模式+面试题汇总+性能调优/架构设计+源码解析
本文是“Java学习路线”专栏的导航文章,目标是为Java初学者和初中高级工程师提供一套完整的Java学习路线。
|
7天前
|
SQL Java 编译器
Java——类与对象(封装)
封装是面向对象编程中的概念,指将数据(属性)和相关操作(方法)组合成独立单元(类),使外部无法直接访问对象的内部状态,只能通过提供的方法进行交互,从而保护数据安全。例如,手机将各种组件封装起来,只暴露必要的接口供外部使用。实现封装时,使用`private`关键字修饰成员变量,并提供`get`和`set`方法进行访问和修改。此外,介绍了包的概念、导入包的方式及其注意事项,以及`static`关键字的使用,包括静态变量和方法的初始化与代码块的加载顺序。
18 10
Java——类与对象(封装)
|
6天前
|
Java
Java——抽象类和接口
抽象类是一种不能被实例化的类,至少包含一个抽象方法(无实现体的方法),常用于定义一组相关类的共同特征,并强制子类实现特定方法。抽象方法不能被 `static` 或 `final` 修饰,且必须被重写。 接口则是一个完全抽象的类,用于规范类的行为。接口使用 `interface` 关键字定义,不能实例化,并且类与接口之间是实现关系。 内部类是在一个类内定义的类,分为成员内部类、静态内部类、局部内部类和匿名内部类。成员内部类可被修饰符修饰,静态内部类只能访问外部类的静态成员,局部内部类定义在方法内,匿名内部类则隐藏了名字,直接通过 `new` 关键字定义并实现接口或继承类。
12 5
Java——抽象类和接口
|
6天前
|
Java
Java——接口的使用实例
Comparable接口用于自定义类的对象比较。通过实现此接口并重写`compareTo`方法,可以定义自定义类型的比较规则。 接下来介绍了Comparator接口,它提供了一种更灵活的比较方式。通过实现Comparator接口并重写`compare`方法,可以根据不同属性定义不同的比较规则。例如,定义一个`BrandComparator`类来比较汽车的品牌。 最后,介绍了Cloneable接口,用于实现对象的克隆。实现该接口并重写`clone`方法后,可以创建对象的浅拷贝或深拷贝。浅拷贝仅复制对象本身,深拷贝则会递归复制所有成员变量。
13 4
Java——接口的使用实例
|
7天前
|
缓存 前端开发 Java
【Java面试题汇总】Spring,SpringBoot,SpringMVC,Mybatis,JavaWeb篇(2023版)
Soring Boot的起步依赖、启动流程、自动装配、常用的注解、Spring MVC的执行流程、对MVC的理解、RestFull风格、为什么service层要写接口、MyBatis的缓存机制、$和#有什么区别、resultType和resultMap区别、cookie和session的区别是什么?session的工作原理
【Java面试题汇总】Spring,SpringBoot,SpringMVC,Mybatis,JavaWeb篇(2023版)
|
12天前
|
Java 数据库连接 数据库
Java服务提供接口(SPI)的设计与应用剖析
Java SPI提供了一种优雅的服务扩展和动态加载机制,使得Java应用程序可以轻松地扩展功能和替换组件。通过合理的设计与应用,SPI可以大大增强Java应用的灵活性和可扩展性。
44 18
|
10天前
|
Java 开发者
Java的接口详解
Java接口是编程中的一种重要特性,用于定义方法签名而不提供具体实现,作为类之间的契约,使不同类能以统一方式交互。接口使用`interface`关键字定义,可包含方法声明和常量。类通过`implements`关键字实现接口,并可同时实现多个接口,解决多重继承问题。接口中的方法默认为抽象方法,变量默认为`public static final`。Java 8引入了默认方法和静态方法,增强接口功能。接口广泛应用于回调机制和多态性实现,有助于构建更灵活和可维护的代码结构。
|
8天前
|
SQL JSON JavaScript
JavaWeb基础9——VUE,Element&整合Javaweb的商品管理系统
Vue 指令、生命周期、this和$、vue脚手架进行模块化开发/ElementUI框架、综合案例,element商品列表展示增删改查
JavaWeb基础9——VUE,Element&整合Javaweb的商品管理系统