SpringBoot中使用Hibernate-validator的自定义validator和GlobalExceptionHandLer进行参数校验和异常处理

前言
虽然hibernate已经支持了很多基于注解的参数校验,但随着业务的复杂,其丰富的注解往往还是无法满足我们一些特定的场景,所以有时需要我们根据业务场景自定义一些注解来实现特定的参数校验。
在项目中也需要对一些异常进行统一的管理,并返回前端特定的message,此时可以使用全局异常进行统一处理
正文

1.创建实体类

public class Request {

    /**
     * 年
     * yyyy
     */
    public interface YearValid{}

    /**
     * 月
     * yyyy-MM
     */
    public interface MonthValid{}

    /**
     * 日
     * yyyy-MM-dd
     */
    public interface DayValid{}

    @StrYear(groups = YearValid.class)
    @StrMonth(groups = MonthValid.class)
    @StrDay(groups = DayValid.class)
    private String queryTime;
}

2.创建上述自定义注解(@StrYear、@StrMonth、@StrDay,三者逻辑相似,此处只实现@StrMonth)

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = {StrMonthValidator.class})//指定该注解校验逻辑
public @interface StrMonth {
    String message() default "请输入正确的查询时间";

    Class<?>[] groups() default {};

    Class<? extends Payload> [] payload() default { };
}

import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;
import java.text.ParseException;
import java.text.SimpleDateFormat;
/**
 * 对传入的 String类型的时间进行校验
 */
public class StrMonthValidator implements ConstraintValidator<StrMonth, String> {

    /**
     * 校验String类型的串是否符合月类型
     *
     * @param value
     * @param context
     * @return
     */
    @Override
    public boolean isValid(String value, ConstraintValidatorContext context) {

        SimpleDateFormat format = null;
        boolean b = true;
        if (null != value && value.replace(" ", "").length() == 7) {
            format = new SimpleDateFormat("yyyy-MM");
            try {
                format.parse(value);
            } catch (ParseException e) {
                b = false;
            }
        } else {
            b = false;
        }
        return b;
    }
}

3.controller层使用@Validated注解对实体属性进行校验
因业务中对于queryTime字段在不同接口中有不同的定义,所以使用了validator的groups分组模式,以便适配不同的情况

/**
 * 校验传入的参数是否为正确的月份
 */
@PostMapping("/test")
public String memberOverallSummary(@RequestBody @Validated(value = RequestVo.MonthValid.class) RequestVo vo) {
    System.out.println(vo.toString());
    return vo.toString();
}

4.到此,已经可以实现参数校验了,但我们还没有对异常进行捕捉,此时,实现自定义异常,下边代码块中的ApiError为字定义的异常实体类

/**
     * 处理所有接口数据验证异常
     *
     * @param e
     * @returns
     */
    @ResponseStatus(HttpStatus.BAD_REQUEST)
    @ExceptionHandler(MethodArgumentNotValidException.class)
    public R handleMethodArgumentNotValidException(MethodArgumentNotValidException e) {
        // 打印堆栈信息
        log.error(e.getMessage(), e);
        String msg = "校验失败: ";
        BindingResult bindingResult = e.getBindingResult();
        for (FieldError error : bindingResult.getFieldErrors()) {
            msg += error.getDefaultMessage() + ";";
        }
        ApiError apiError = new ApiError(BAD_REQUEST.value(), msg);
        return buildResponseEntity(apiError);
    }

5.此时访问我们的接口,已经实现了参数校验,并且可以返回友好的异常提示信息
对于入参的queryTime参数我们不传或者是传空,或者传入的数据不符合时间格式,都会抛出我们希望的异常

{
    "code": 400,
    "msg": "校验失败: 请输入正确的查询时间;",
    "data": null,
    "success": false
}

PS:最后,如果我们有多个异常,希望校验器校验到第一个参数异常时立即返回,可以配置fail-fast,配置如下

/**
 * 参数校验 fast-fail配置类
 */
@Configuration
public class ValidatorConfig {

    @Bean
    public Validator validator(){
        ValidatorFactory validatorFactory = Validation.byProvider( HibernateValidator.class )
                .configure() // true-快速失败返回模式    false-普通模式
                .addProperty( "hibernate.validator.fail_fast", "true" )
                .buildValidatorFactory();
        Validator validator = validatorFactory.getValidator();

        return validator;
    }
}

至此,我们已经可以优雅的实现参数校验和异常处理了

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Hibernate Validator是一个用于Java Bean验证的框架,它提供了一组注解和API,用于验证Java Bean的属性值是否符合指定的规则和约束。它可以用于验证表单数据、REST API请求、数据库实体等各种场景下的数据。Hibernate Validator支持多种验证规则,包括基本数据类型、字符串、日期、集合、数组等。它还支持自定义验证规则和错误消息。使用Hibernate Validator可以有效地减少代码量和提高开发效率。 ### 回答2: Hibernate Validator是一个流行的Java开源校验框架,它是基于JSR 303规范(Bean验证)的实现。它提供了一组注释和API,用于对JavaBean进行验证和校验。 Hibernate Validator提供了很多内置的校验注释,比如@NotNull、@NotEmpty、@Min、@Max等。这些注释可以直接应用在JavaBean的属性上,通过注释指定的校验规则来验证属性的值是否合法。同时,Hibernate Validator还支持自定义校验注释,可以根据业务需求来定义新的注释,并实现相应的校验逻辑。 Hibernate Validator不仅可以对单个属性进行校验,还支持对整个JavaBean进行校验。例如,可以在JavaBean的类级别上添加@Valid注释,来验证整个对象的合法性。此外,还可以通过分组校验来实现在不同场景下不同的校验规则。 除了注释方式外,Hibernate Validator还提供了一套强大的API,通过编程方式来进行校验和验证。通过ValidatorFactory和Validator两个主要的接口,可以创建Validator对象并执行校验操作。可以验证单个属性的值,也可以验证整个JavaBean对象。 Hibernate Validator还提供了国际化的支持,可以根据不同的区域设置显示不同的错误信息。同时,还能够将校验错误信息与具体的校验注释进行关联,方便开发者快速定位问题。 总结来说,Hibernate Validator提供了一种有效、灵活和方便的方式来对JavaBean进行校验。它的注释和API丰富多样,并且支持自定义校验规则,同时还提供了国际化和错误信息关联等特性,使得校验过程更加强大和可控。它在Java开发的应用越来越广泛,为开发者提供了一种便捷的校验解决方案。 ### 回答3: Hibernate Validator是一个基于JSR 380规范的校验框架,它可以轻松地对Java对象进行校验。通过使用Hibernate Validator,开发人员可以在应用程序方便地添加校验规则,并且可以验证这些规则是否被满足。 Hibernate Validator提供了一组注解,这些注解可以附加在JavaBean的属性上,以标识需要进行校验的规则。例如,@NotNull注解用于确保属性的值不为空,@Size注解用于确保字符串类型的属性的长度在指定范围内,等等。除了注解外,Hibernate Validator还提供了一些内置的校验器,用于验证各种数据类型的属性,例如字符串、数字、日期等。 使用Hibernate Validator进行校验非常简单。只需要在需要校验的JavaBean上添加注解,并在需要校验的时候调用校验方法即可。校验方法会返回一个包含校验结果的对象,开发人员可以根据需要进行处理。校验方法还可以接受一个可选的校验分组参数,用于校验不同场景下的不同规则。 Hibernate Validator还提供了一些扩展功能,用于自定义校验规则。开发人员可以创建自定义的校验注解,并编写相应的校验器来实现特定的校验逻辑。这使得Hibernate Validator非常灵活,可以满足各种不同的校验需求。 总结来说,Hibernate Validator是一个强大而灵活的校验框架,能够方便地对Java对象进行校验。使用Hibernate Validator可以增加应用程序的稳定性和可靠性,减少错误和异常的发生。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值