• Java中通过反射+自定义注解判断对象中部分属性是否为空,返回为空字段的名称或自定义含义


    场景

    若依管理系统前后端分离版基于ElementUI和SpringBoot怎样实现Excel导入和导出:

    若依管理系统前后端分离版基于ElementUI和SpringBoot怎样实现Excel导入和导出_霸道流氓气质的博客-CSDN博客

    在上面进行excel导入时,需要对数据进行非空校验。

    比如对象中有多个属性,需要判断其中几个属性是否为空,为空则返回给前端提示。新建实体类,有如下字段

    1. @Data
    2. @AllArgsConstructor
    3. @NoArgsConstructor
    4. @Builder
    5. public class LimitQuotaStatistics extends BaseEntity
    6. {
    7.     private static final long serialVersionUID = 1L;
    8.     /** id */
    9.     private Long id;
    10.     /** 部门id */
    11.     private Long deptId;
    12.     /** 部门名称 */
    13.     @Excel(name = "部门名称")
    14.     private String deptName;
    15.     /** 夜班人数 */
    16.     @Excel(name = "夜班人数")
    17.     private Long nightShiftNum;
    18.     /** 早班人数 */
    19.     @Excel(name = "早班人数")
    20.     private Long morningShiftNum;
    21.     /** 早中班人数 */
    22.     @Excel(name = "早中班人数")
    23.     private Long morningMiddlleShiftNum;
    24.     /** 晚中班人数 */
    25.     @Excel(name = "晚中班人数")
    26.     private Long nightMiddleShiftNum;
    27.     /** 合计人数 */
    28.     @Excel(name = "合计人数")
    29.     private Long totalNum;
    30.     /** 定额人数 */
    31.     @Excel(name = "定额人数")
    32.     private Long quotaNum;
    33.     /** 计划日期 */
    34.     @JsonFormat(pattern = "yyyy-MM-dd")
    35.     private Date planDate;
    36. }

    这里的@Excel注解为若依自带的自定义注解,下面会用到其name属性。

    注:

    博客:
    霸道流氓气质的博客_CSDN博客-C#,架构之路,SpringBoot领域博主
    关注公众号
    霸道的程序猿
    获取编程相关电子书、教程推送与免费下载。

    实现

    1、构造一个实体对象,并且只给morningShiftNum赋值

    1.         LimitQuotaStatistics limitQuotaStatistics = LimitQuotaStatistics.builder()
    2.                 .morningShiftNum(3l)
    3.                 .build();

    2、假设我们需要对其中的某些字段进行非空校验

    把需要进行非空校验的字段抽离到Constants

    1. public class LimitConstants
    2. {
    3.     public static final List<String> LIMIT_QUOTA_IMPORT_NOTNULL_CHECK = new ArrayList<String>(){{
    4.         this.add("nightShiftNum");
    5.         this.add("morningShiftNum");
    6.         this.add("morningMiddlleShiftNum");
    7.         this.add("totalNum");
    8.         this.add("quotaNum");
    9.     }};
    10. }

    假如需要对上面这几个字段进行非空校验,如果这几个字段中有为null,则返回对应的字段名

    那么上面构造的实体对象,应该返回nightShiftNum、morningMiddlleShiftNum、totalNum、quotaNum这四个字段。

    3、封装一个工具类方法,通过反射获取该对象的所有字段,然后筛选字段是否在上面要检查的list中并且值为null的即可

    1.     /**
    2.      * 检查对象的属性是否为空
    3.      * @param obj 检查对象
    4.      * @param checkField 要检查的属性名list
    5.      * @return 为空的属性名list
    6.      * @throws IllegalAccessException
    7.      */
    8.     public static List<String> checkObjFieldIsNull(Object obj, List<String> checkField) throws IllegalAccessException {
    9.         List<String> res = new ArrayList<>();
    10.         //获取对象的类
    11.         Class<?> aClass = obj.getClass();
    12.         //获取对象的所有字段,包含public、private和proteced
    13.         Field[] declaredFields = aClass.getDeclaredFields();
    14.         //筛选出包含在要检查的list 并且 属性为null的字段
    15.         Arrays.stream(declaredFields).forEach(field -> {
    16.             //设置属性可访问
    17.             field.setAccessible(true);
    18.             try {
    19.                 if(checkField.contains(field.getName()) && field.get(obj) == null){
    20.                     //获取字段名
    21.                     res.add(field.getName());
    22.                 }
    23.             } catch (IllegalAccessException e) {
    24.                 e.printStackTrace();
    25.             }
    26.         });
    27.         return res;
    28.     }

    然后调用进行测试

    1.     @Test
    2.     public void checkFieldIsNull(){
    3.         LimitQuotaStatistics limitQuotaStatistics = LimitQuotaStatistics.builder()
    4.                 .morningShiftNum(3l)
    5.                 .build();
    6.         List<String> checkFields = LimitConstants.LIMIT_QUOTA_IMPORT_NOTNULL_CHECK;
    7.         try {
    8.             checkObjFieldIsNull(limitQuotaStatistics,checkFields).stream().forEach(System.out::println);
    9.         } catch (IllegalAccessException e) {
    10.             e.printStackTrace();
    11.         }
    12.     }

    测试结果

    但是把这些对象的字段名直接返回给前端提示不大友好,是否可以直接返回对应的自定义的含义名称。

    4、回到上面自定义注解@Excel中,是否可以获取每个字段上的name的值

    首先自定义注解,比如这里的@Excel,就可以将注解作用在类上

    1. /**
    2.  * 自定义导出Excel数据注解
    3.  *
    4.  * @author ruoyi
    5.  */
    6. @Retention(RetentionPolicy.RUNTIME)
    7. @Target(ElementType.FIELD)
    8. public @interface Excel
    9. {
    10.     /**
    11.      * 导出时在excel中排序
    12.      */
    13.     public int sort() default Integer.MAX_VALUE;
    14.     /**
    15.      * 导出到Excel中的名字.
    16.      */
    17.     public String name() default "";
    18. }

    这里只做演示用,所以只保留了一个name属性

    @Retention(RetentionPolicy.RUNTIME)表示注解不仅被保存到class文件中,jvm加载class文件之后,仍然存在。

    @Target(ElementType.FIELD)表示此注解的作用目标是字段和枚举的常量上

    2、创建实体类,并添加上面的自定义注解

    1. @Data
    2. @AllArgsConstructor
    3. @NoArgsConstructor
    4. @Builder
    5. public class LimitQuotaStatistics extends BaseEntity
    6. {
    7.     private static final long serialVersionUID = 1L;
    8.     /** id */
    9.     private Long id;
    10.     /** 部门id */
    11.     private Long deptId;
    12.     /** 部门名称 */
    13.     @Excel(name = "部门名称")
    14.     private String deptName;
    15.     /** 夜班人数 */
    16.     @Excel(name = "夜班人数")
    17.     private Long nightShiftNum;
    18.     /** 早班人数 */
    19.     @Excel(name = "早班人数")
    20.     private Long morningShiftNum;
    21.     /** 早中班人数 */
    22.     @Excel(name = "早中班人数")
    23.     private Long morningMiddlleShiftNum;
    24.     /** 晚中班人数 */
    25.     @Excel(name = "晚中班人数")
    26.     private Long nightMiddleShiftNum;
    27.     /** 合计人数 */
    28.     @Excel(name = "合计人数")
    29.     private Long totalNum;
    30.     /** 定额人数 */
    31.     @Excel(name = "定额人数")
    32.     private Long quotaNum;
    33.     /** 计划日期 */
    34.     @JsonFormat(pattern = "yyyy-MM-dd")
    35.     private Date planDate;
    36. }

    3、通过反射获取对象属性的注解以及注解的值

    field.getAnnotation(Excel.class).name()

    具体示例:

    1.     /**
    2.      * 检查对象的属性是否为空
    3.      * @param obj 检查对象
    4.      * @param checkField 要检查的属性名list
    5.      * @return 为空的属性名list
    6.      * @throws IllegalAccessException
    7.      */
    8.     public static List<String> checkObjFieldIsNull(Object obj, List<String> checkField) throws IllegalAccessException {
    9.         List<String> res = new ArrayList<>();
    10.         //获取对象的类
    11.         Class<?> aClass = obj.getClass();
    12.         //获取对象的所有字段,包含public、private和proteced
    13.         Field[] declaredFields = aClass.getDeclaredFields();
    14.         //筛选出包含在要检查的list 并且 属性为null的字段
    15.         Arrays.stream(declaredFields).forEach(field -> {
    16.             //设置属性可访问
    17.             field.setAccessible(true);
    18.             try {
    19.                 if(checkField.contains(field.getName()) && field.get(obj) == null){
    20.                     //获取自定义注解标识的属性名 如果没有则忽略
    21.                     res.add(field.getAnnotation(Excel.class).name()); 
    22.                 }
    23.             } catch (IllegalAccessException e) {
    24.                 e.printStackTrace();
    25.             }
    26.         });
    27.         return res;
    28.     }

    测试结果

    ​​​​​​​

  • 相关阅读:
    【无标题】
    强大而灵活的python装饰器
    ABAP读取销售订单选配BOM函数-CS_BOM_EXPL_KND_V1
    OCP Java17 SE Developers 复习题06
    c语言变长数组的实现
    吴恩达deeplearning.ai:决策树模型
    QT 知:qmake 手册
    git总结
    ES6 Iterator 和 for...of 循环
    优维低代码:Redirect 路由重定向&If 条件渲染
  • 原文地址:https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/details/127847218