【自定义注解实现字段检验重复值】


前言


一、业务场景?

对于一些数据库表字段是唯一索引的,需要在数据插入前做数据重复性检验,并友好返回给前端。例如用户名不能重复,工号不能重复等。对于这些重复判断项目中很多地方都需要用到,想法是直接抽离处理,统一管理,减少重复代码。按照以下步骤配置,字段值重复检验一个注解即可实现。以下是实现步骤,如有疑问,欢迎大家提问和指点。

二、实现步骤

1.引入依赖

代码如下(示例):

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-validation</artifactId>
    <version>2.3.11.RELEASE</version>
</dependency>

2.新增自定义注解(注解嵌套)

@FieldRepeatValidators

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = FieldRepeatValidatorHandler.class)
public @interface FieldRepeatValidators {

    /**
     * 数据表名
     * @return
     */
    String tableName();

    /**
     * 数据表id字段名
     * @return
     */
    String idField() default "id";
    
     /**
     * id字段的属性名称
     * @return
     */
    String idProperty() default "id";
    

    FieldRepeatValidator[] fieldRepeatValidators();

    String message() default "";
    
    Class<?>[] groups() default { };
    
    Class<? extends Payload>[] payload() default { };
}

@FieldRepeatValidator

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.PARAMETER, ElementType.FIELD, ElementType.TYPE})
public @interface FieldRepeatValidator {

    /**
     * 注解属性 - 对应校验字段,表字段
     * @return
     */
    String field();

    /**
     * 默认错误提示信息
     * @return
     */
    String message() default "字段内容重复!";

    /**
     *
     * @return 实体类属性
     */
    String property();

}

3. 自定义注解类处理器

public class FieldRepeatValidatorHandler implements ConstraintValidator<FieldRepeatValidators, Object>, ApplicationContextAware {

    private FieldRepeatValidators fieldRepeatValidators;

    static ApplicationContext application;

    @Override
    @SuppressWarnings("all")
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        application = applicationContext;
    }

    @Override
    public void initialize(FieldRepeatValidators fieldRepeatValidators) {
        this.fieldRepeatValidators = fieldRepeatValidators;
    }

    @Override
    public boolean isValid(Object o, ConstraintValidatorContext constraintValidatorContext) {
        return fieldRepeat(o,fieldRepeatValidators);
    }

    public static boolean fieldRepeat(Object object, FieldRepeatValidators fieldRepeatValidators) {
        JSONObject jsonObject = JSONUtil.parseObj(object);
        String table = fieldRepeatValidators.tableName();
        boolean isUpdate;
        String idField = fieldRepeatValidators.idField();
        String idProperty = fieldRepeatValidators.idProperty();
        isUpdate = jsonObject.containsKey(idProperty);
        Object id = jsonObject.get(idProperty);
        HashMap<String, Object> errors = new HashMap<>();
        // 逐一将该表需要判断重复逻辑的字段进行读取并分别处理
        for(FieldRepeatValidator fieldRepeatValidator : fieldRepeatValidators.fieldRepeatValidators()){
            String field = fieldRepeatValidator.field();
            String property = fieldRepeatValidator.property();
            String message = fieldRepeatValidator.message();
            Object o = jsonObject.get(property);
            if(Objects.isNull(o)){
                continue;
            }
    		// 新增操作只需判断表中是否有重复字段的记录
            String sql = StrUtil.format("select count(1) 'count' from {} where {} = '{}'",table,field,o);
            // 更新操作需要排除自身再去判断是否有其他重复字段的记录
            if(isUpdate){
                sql = StrUtil.format(sql + " and {} <> {}",idField,id);
            }
            JdbcTemplate jdbcTemplate = application.getBean("jdbcTemplate", JdbcTemplate.class);
             Map<String, Object> map = jdbcTemplate.queryForMap(sql);
            if(!map.get("count").equals(0L)){
            	// 将重复字段异常全部汇总到异常表返回到前端
                errors.put(property,message);
            }
        }
        if(errors.size() != 0){
            throw new FieldRepeatException(errors);
        }
        return true;
    }
}

4. 自定义异常类

@Data
public class FieldRepeatException extends ConstraintDeclarationException {

    /**
     *
     * @param cause
     */
    private HashMap<String,Object> errors;
    
    public FieldRepeatException(HashMap<String,Object> errors) {
        this.errors = errors;
    }
}

5. 全局异常捕获类

@RestControllerAdvice()
@Slf4j
public class GlobalControllerExceptionHandler {
    
    @ExceptionHandler(FieldRepeatException.class)
    private R fieldRepeatExceptionHandler(FieldRepeatException e) {
        return R.error("40001","数据检验失败").put("error",e.getErrors()));
    }
}

6. 实体类上的注解

@Data
@EqualsAndHashCode(callSuper = true)
@ApiModel(value = "User对象", description = "用户表")
@TableName("user")
@FieldRepeatValidators(tableName = "user",idField = "id",fieldRepeatValidators = {
        @FieldRepeatValidator(field = "user_name", property = "userName",message = "用户名重复!"),
        @FieldRepeatValidator(field = "mobile", property = "mobile",message = "电话号码重复!")
})
public class User extends Model<User> {
    
    @TableId(value = "id", type = IdType.AUTO)
    private Integer id;
      /**
       * 用户名
       */
    @ApiModelProperty(value = "用户名")
    private String userName;
    
      /**
       * 电话
       */
    @ApiModelProperty(value = "电话")
    private String mobile;
    
}

7. Controller层

@PostMapping("/save")
@ApiOperation(value = "新增", notes = "传入user")
public R save(@RequestBody @Validated User user) {
    // 增加@Validated注解
   return R.data(userService.save(user));
}

结果显示

{
    "code":40001,
    "message":"数据检验失败",
    "errors":{
        "userName":"用户名重复!",
        "mobile":"电话号码重复!"    
    }
}

三、踩坑日记

问题:
检验完成,抛出自定义异常会出现: unexpected exception during isValid call
原因:
自定义异常由于继承的是RunTimeException,导致在检验处理类执行isValid()方法时没有正常返回值,而抛出了javax.validation.ValidationException。导致自定义异常无法被全局异常处理类捕获。
解决方法:
自定义异常需要继承ConstraintDeclarationException

public class FieldRepeatException extends ConstraintDeclarationException{}

问题:
javax.validation.ConstraintDefinitionException: HV000074:containsbut does not contain a message parameter
原因:
自定义检验字段注解缺少message参数
解决方法:
添加message参数等

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = FieldRepeatValidatorHandler.class)
public @interface FieldRepeatValidators {
    
    String message() default "";
    Class<?>[] groups() default { };
    Class<? extends Payload>[] payload() default { };

}

问题:
自定义检验字段注解添加在字段上,只能拿到该字段的当前值。这样的话就不能获取到其他字段的值,其中最重要的是id字段的值。id字段值的作用是判断此次请求是新增还是修改,如果是新增,则只需要判断数据库中是否含有该字段值的记录。如果是修改,则需要判断除了自身id记录外是否还含有该字段值的记录。
解决方法:
自定义检验字段注解在类上,则可以获取到全部字段的值。再通过注解逐一对有需要做重复检验的字段进行检验,返回检验结果。

@FieldRepeatValidators(tableName = "user",idField = "id",fieldRepeatValidators = {
        @FieldRepeatValidator(field = "user_name", property = "userName",message = "用户名重复!"),
        @FieldRepeatValidator(field = "mobile", property = "mobile",message = "电话号码重复!")
})
public class User extends Model<User> {
    
}

四、参考博客

https://blog.csdn.net/weixin_44912855/article/details/122720171
https://juejin.cn/post/6844903950768930830#heading-14

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
Java EE常用框架 WebService 介绍 基于Web的服务。它使用Web(HTTP)方式,接收和响应外部系统的某种请求。从而实现远程调用 术语 XML. Extensible Markup Language -扩展性标记语言 WSDL – WebService Description Language – Web服务描述语言。 SOAP-Simple Object Access Protocol(简单对象访问协议) SOA(Service-Oriented Architecture) :面向服务的架构 它是一种思想,IBM大力倡导是即插即用的,IBM大力提倡,希望以组装电脑的方式来开发应用 它是目录服务,通过该服务可以注册和发布webservcie,以便第三方的调用者统一调用 使用: 二、我们可以使用Java自带的WsImport来实现本地代理。这种方法会将WebService翻译成Java类,我们使用类一样去访问WebService就行了。非常好用。 三、除了调用别人发布的webService,也可以自己发布WebService服务 四、CXF框架可以与spring无缝连接,就不用我们自己Endpoint了。它还能记录日志之类的 五、我们还可以使用Idea下的webservice,能够使用图形画面的方式获取本地代理和生成WSDL文件。 Activiti 介绍 Activiti5是一个业务流程管理(BPM)框架 如果我们的业务是比较复杂的话,我们才会用到工作流! 使用Activiti的步骤 定义工作流 画一个BPMN图就可以了 部署工作流 执行工作流-->按照我们定义的工作流来执行 工作流在执行的过程中肯定会涉及到很多数据,Activiti是默认提供数据库表给我们使用的 Activiti工作流框架快速入门: 定义工作流,使用插件来把我们的流程图画出来。这个流程图就是我们定义的工作流。 工作流引擎是工作流的核心,能够让我们定义出来的工作流部署起来。 由于我们使用工作流的时候是有很多数据产生的,因此Activiti是将数据保存到数据库表中的。这些数据库表由Actitviti创建,由Activiti维护。 部署完的工作流是需要手动去执行该工作流的。 根据由谁处理当前任务,我们就可以查询出具体的任务信息。 根据任务的id,我们就可以执行任务了。 细节 流程定义:涉及到了四张数据库表 我们可以通过API把我们的流程定义图读取出来 可以根据查询最新版本的流程定义 删除流程定义 部署流程定义的时候也可以是ZIP文件 流程运行:涉及到两个对象,四张数据库表: 流程实例 获取流程实例和任务的历史信息 判断流程实例是否为空来判断流程是否结束了 查看正在运行服务的详细信息 通过流程实例来开启流程 流程变量:它涉及到了两张表。 流 程变量实际上就是我们的条件。 作用 我们可以在流程开始的时候设置流程变量,在任务完成的时候设置流程变量。 运行时服务和流程任务都可以设置流程变量。 连线 通过连线我们可以在其中设置条件,根据不同的条件流程走不同的分支 排他网关 SpringData JPA 简介 API Repository接口 PagingAndSortingRepository JpaRepository JpaSpecificationExecutor 过滤条件查询接口 注解 nameQuery注解 SQL命名,调用的时候根据名称调用 查询注解 1,targetEntity 属性表示默认关联的实体类型,默认为当前标注的实体类。 2,cascade属性表示与此实体一对一关联的实体的级联样式类型。 3,fetch属性是该实体的加载方式,默认为即时加载EAGER 4,optional属性表示关联的该实体是否能够存在null,默认为ture,如果设置为false,则该实体不能为null, 5, mapperBy属性:指关系被维护端 1,@JoinColumn注释是保存表与表之间关系的字段 2,如果不设置name,默认name = 关联表的名称+”-“+关联表

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值