欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

SpringBoot+Hibernate實現(xiàn)自定義數(shù)據(jù)驗證及異常處理

 更新時間:2022年04月22日 16:00:06   作者:花傷情猶在  
這篇文章主要為大家介紹了SpringBoot如何整合Hibernate自定義數(shù)據(jù)驗證及多種方式異常處理,文中的示例代碼講解詳細,感興趣的可以了解一下

前言

在進行 SpringBoot 項目開發(fā)中,經(jīng)常會碰到屬性合法性問題,而面對這個問題通常的解決辦法就是通過大量的 if 和 else 判斷來解決的,例如:

@PostMapping("/verify")
    @ResponseBody
    public Object verify(@Valid User user){
        if (StringUtils.isEmpty(user.getName())){
            return "姓名不能為空";
        }
        if (StringUtils.isEmpty(user.getAge())){
            return "姓名不能為空";
        }
        if (!StringUtils.isEmpty(user.getSex())&&user.getSex().equals("男")&&user.getSex().equals("女")){
            return "性別有誤";
        }
        return user;
    }

這種代碼寫法十分麻煩,試想一下如果你有10個、20個字段屬性,你也要跟著寫十幾二十幾個 if 和 else 判斷?

So,本文講解一下使用Hibernate框架來去驗證字段屬性,使用相應的注解即可實現(xiàn)字段合法性校驗,以及如何自定義注解進行校驗,包括出現(xiàn)異常的幾種處理方式。

Hibernate實現(xiàn)字段校驗

Maven依賴

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

常用的校驗注解

注解釋義
@Null必須為null
@NotNull不能為null
@AssertTrue必須為true
@AssertFalse必須為false
@Min必須為數(shù)字,其值大于或等于指定的最小值
@Max必須為數(shù)字,其值小于或等于指定的最大值
@DecimalMin必須為數(shù)字,其值大于或等于指定的最小值
@DecimalMax必須為數(shù)字,其值小于或等于指定的最大值
@Size集合的長度
@Digits必須為數(shù)字,其值必須再可接受的范圍內(nèi)
@Past必須是過去的日期
@Future必須是將來的日期
@Pattern必須符合正則表達式
@Email必須是郵箱格式
@Length長度范圍
@NotEmpty不能為null,長度大于0
@Range元素的大小范圍
@NotBlank不能為null,字符串長度大于0(限字符串)

定義User實體類

@Data
public class User {
    @NotBlank(message = "姓名不能為空")
    private String name;
    @NotBlank(message = "年齡不能為空")
    private String age;
}

定義UserController

@Controller
public class UserController {

    @PostMapping("/verify")
    @ResponseBody
    public Object verify(@Valid User user, BindingResult result){
    	//字段校驗有錯誤
        if (result.hasErrors()){
        	//獲取錯誤字段信息
            List<FieldError> fieldErrors = result.getFieldErrors();
            if (fieldErrors!=null){
            	//創(chuàng)建一個map用來封裝字段錯誤信息
                HashMap<String, String> map = new HashMap<>();
                //遍歷錯誤字段
                fieldErrors.forEach(x->{
                	//獲取字段名稱
                    String field = x.getField();
                    //獲取字段錯誤提示信息
                    String msg = x.getDefaultMessage();
                    //存入map
                    map.put(field, msg);
                });
                return map;
            }
        }
        return user;
    }

}

啟動項目進行測試

可以看到name和age的錯誤信息已經(jīng)封裝好傳回來了

自定義校驗注解

自定義一個校驗性別的注解Sex

/**
 * 性別約束
 * @Target用于指定使用范圍,該處限定只能在字段上使用
 * @Retention(RetentionPolicy.RUNTIME)表示注解在運行時可以通過反射獲取到
 * @Constraint(validatedBy = xxx.class)指定該注解校驗邏輯
 */
@Target({ ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Constraint(validatedBy = SexConstraintValidator.class)
public @interface Sex {

    String message() default "性別有誤";

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

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

創(chuàng)建SexConstraintValidator校驗邏輯類

/**
 * 性別約束邏輯判斷
 */
public class SexConstraintValidator implements ConstraintValidator<Sex, String> {

    @Override
    public boolean isValid(String value, ConstraintValidatorContext context) {
        return value != null && (value.equals("男") || value.equals("女"));
    }

}

修改User實體類

@Data
public class User {
    @NotBlank(message = "姓名不能為空")
    private String name;
    @NotBlank(message = "年齡不能為空")
    private String age;
    @Sex(message = "性別不能為空或有誤")
    private String sex;
}

重啟項目測試效果

使用AOP處理校驗異常

Maven依賴

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-aop</artifactId>
</dependency>

這里我們用注解作為AOP的切入點,新建一個注解 BindingResultAnnotation

@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface BindingResultAnnotation {
}

定義參數(shù)校驗切面類

/**
 * 參數(shù)校驗切面類
 */
@Aspect
@Component
public class BindingResultAspect {

    /**
     * 校驗切入點
     */
    @Pointcut("@annotation(com.hsqyz.hibernate.config.aop.BindingResultAnnotation)")
    public void BindingResult() {
    }

    /**
     * 環(huán)繞通知
     * @param joinPoint
     * @return
     * @throws Throwable
     */
    @Around("BindingResult()")
    public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable {
        System.out.println("參數(shù)校驗切面...");
        Object[] args = joinPoint.getArgs();
        for (Object arg : args) {
            if (arg instanceof BindingResult) {
                BindingResult result = (BindingResult) arg;
                if (result.hasErrors()){
                    List<FieldError> fieldErrors = result.getFieldErrors();
                    if (fieldErrors!=null){
                        HashMap<String, String> map = new HashMap<>();
                        fieldErrors.forEach(x->{
                            String field = x.getField();
                            String msg = x.getDefaultMessage();
                            map.put(field, msg);
                        });
                        return map;
                    }
                }

            }
        }
        return joinPoint.proceed();
    }


}

修改UserController

注意:這里將新建的切面注解添加到方法上@BindingResultAnnotation,必須攜帶BindingResult result在參數(shù)后面,否則AOP無法獲取錯誤信息,導致AOP無法處理異常。

@Controller
public class UserController {

    @PostMapping("/verify")
    @ResponseBody
    @BindingResultAnnotation
    public Object verify(@Valid User user, BindingResult result) {
        return user;
    }

}

重啟項目查看效果

全局異常類處理異常

創(chuàng)建GlobelExceptionHandler來處理全局異常,使用@ExceptionHandle來攔截指定異常,由于參數(shù)校驗拋出的異常是BindException,所以我們需要攔截BindException異常,而BindException內(nèi)部封裝這錯誤信息,這樣就可以用全局異常處理類來封裝字段錯誤信息返回。

/**
 * 全局異常處理
 */
@ControllerAdvice
public class GlobelExceptionHandler {

    /**
     * 參數(shù)驗證異常處理
     * @param result
     * @return
     */
    @ResponseBody
    @ExceptionHandler(BindException.class)
    public Object bindExceptionHandler(BindingResult result) {
        System.out.println("參數(shù)驗證異常處理...");
        if (result.hasErrors()){
            List<FieldError> fieldErrors = result.getFieldErrors();
            if (fieldErrors!=null){
                HashMap<String, String> map = new HashMap<>();
                fieldErrors.forEach(x->{
                    String field = x.getField();
                    String msg = x.getDefaultMessage();
                    map.put(field, msg);
                });
                return map;
            }
        }
        return result.getAllErrors();
    }


}

修改UserController

注意:這個時候我們就需要去掉verify()方法中的BindingResult result參數(shù),因為不去掉的話,出現(xiàn)錯誤信息不會拋出異常,會被收集起來封裝到BindingResult中去,所以要想使用全局異常處理類來處理校驗異常,就必須去掉BindingResult參數(shù),讓其拋出異常,我們再使用全局異常處理類進行異常處理,封裝異常信息并返回。

@Controller
public class UserController {

    @PostMapping("/verify")
    @ResponseBody
    public Object verify(@Valid User user) {
        return user;
    }

}

重啟項目查看運行效果

到此這篇關于SpringBoot+Hibernate實現(xiàn)自定義數(shù)據(jù)驗證及異常處理的文章就介紹到這了,更多相關SpringBoot Hibernate數(shù)據(jù)驗證 異常處理內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

最新評論