美文网首页Spring BootSpringFrameworkJava学习笔记
Spring boot使用总结(三)校验

Spring boot使用总结(三)校验

作者: joyhj | 来源:发表于2016-08-28 14:36 被阅读7141次

    spring boot 1.4默认使用 hibernate validator 5.2.4 Final实现校验功能。hibernate validator 5.2.4 Final是JSR 349 Bean Validation 1.1的具体实现。
    一 初步使用
    hibernate vilidator主要使用注解的方式对bean进行校验,初步的例子如下所示:

    package com.learn.validate.domain;
    import javax.validation.constraints.Min;
    import org.hibernate.validator.constraints.NotBlank;
    public class Student {
    //在需要校验的字段上指定约束条件
     @NotBlank
     private String name;
     @Min(3)
     private int age;
     @NotBlank
     private String classess;
    
      public String getName() {
      return name;
     }
     public void setName(String name) {
      this.name = name;
     }
     public int getAge() {
      return age;
     }
     public void setAge(int age) {
      this.age = age;
     }
     public String getClassess() {
      return classess;
     }
     public void setClassess(String classess) {
      this.classess = classess;
     }
     
    }
    

    然后在controller中可以这样调用,加上@Validated注解即可。

    package com.learn.validate.controller;
    import org.springframework.validation.annotation.Validated;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RestController;
    
    import com.learn.validate.domain.Student;
    
    @RestController
    public class ValidateController {
    
     @RequestMapping(value="testStudent")
     public void testStudent(@Validated Student student) {
     }
    }
    

    如果校验失败,默认会返回Spring boot 框架的出错信息。是一个json串,里面有详细的出错描述。

    二 使用gruops 属性来实现区别不同的校验需求
    在上面的例子中,如果Student bean想要用于两个不同的请求中,每个请求有不同的校验需求,例如一个请求只需要校验name字段,一个请求需要校验name和age两个字段,那该怎么做呢?
    使用注解的groups属性可以很好的解决这个问题,如下所示:

    package com.learn.validate.domain;
    import javax.validation.constraints.Min;
    
    import org.hibernate.validator.constraints.NotBlank;
    
    public class Student {
        //使用groups属性来给分组命名,然后在需要的地方指定命令即可
        @NotBlank(groups=NAME.class)
        private String name;
        @Min(value=3,groups=AGE.class)
        private int age;
        @NotBlank
        private String classess;
        
        public String getName() {
            return name;
        }
        public void setName(String name) {
            this.name = name;
        }
        public int getAge() {
            return age;
        }
        public void setAge(int age) {
            this.age = age;
        }
        public String getClassess() {
            return classess;
        }
        public void setClassess(String classess) {
            this.classess = classess;
        }
        
        public interface NAME{};
        
        public interface AGE{};
    
    }
    

    根据需要在@Validated属性中指定需要校验的分组名,可以指定1到多个。指定到的分组名会全部进行校验,不指定的不校验。

    package com.learn.validate.controller;
    
    import org.springframework.validation.annotation.Validated;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RestController;
    
    import com.learn.validate.domain.Student;
    import com.learn.validate.domain.Student.AGE;
    import com.learn.validate.domain.Student.NAME;
    
    @RestController
    public class ValidateController {
    
        @RequestMapping(value="testStudent")
        public void testStudent(@Validated Student student) {
            
        }
        
        
        @RequestMapping(value="testStudent1")
        public void testStudent1(@Validated(NAME.class) Student student) {
            
        }
        
        @RequestMapping(value="testStudent2")
        public void testStudent2(@Validated({NAME.class,AGE.class}) 
        Student student) {
            
        }
    }
    

    三 使用 @ScriptAssert 注解校验复杂的业务逻辑
    如果需要校验的业务逻辑比较复杂,简单的@NotBlank,@Min注解已经无法满足需求了,这时可以使用@ScriptAssert来指定进行校验的方法,通过方法来进行复杂业务逻辑的校验,然后返回 true或false来表明是否校验成功。例如下面的例子:

    package com.learn.validate.domain;
    import javax.validation.constraints.Min;
    
    import org.hibernate.validator.constraints.NotBlank;
    import org.hibernate.validator.constraints.ScriptAssert;
    
    import com.learn.validate.domain.Student.CHECK;
    //通过script 属性指定进行校验的方法,传递校验的参数,
    //依然可以通过groups属性指定分组名称
    @ScriptAssert(lang="javascript",script="com.learn.validate.domain
    .Student.checkParams(_this.name,_this.age,_this.classes)",
    groups=CHECK.class)
    public class Student {
     
     @NotBlank(groups=NAME.class)
     private String name;
     @Min(value=3,groups=AGE.class)
     private int age;
     @NotBlank
     private String classess;
     
     public String getName() {
      return name;
     }
     public void setName(String name) {
      this.name = name;
     }
     public int getAge() {
      return age;
     }
     public void setAge(int age) {
      this.age = age;
     }
     public String getClassess() {
      return classess;
     }
     public void setClassess(String classess) {
      this.classess = classess;
     }
     
     public interface NAME{};
     
     public interface AGE{};
     
     public interface CHECK{};
     
     //注意进行校验的方法要写成静态方法,否则会出现 
     //TypeError: xxx is not a function 的错误
     public static boolean checkParams(String name,int age,String classes) {
      if(name!=null&&age>8&classes!=null)
      {
       return true;
      }
      else
      {
       return false;
      }
      
     }
    
    }
    
    

    在需要的地方,通过分组名称进行调用

    package com.learn.validate.controller;
    
    import org.springframework.validation.annotation.Validated;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RestController;
    
    import com.learn.validate.domain.Student;
    import com.learn.validate.domain.Student.CHECK;
    
    @RestController
    public class ValidateController {
     
     @RequestMapping(value="testStudent3")
     public void testStudent3(@Validated(CHECK.class) Student student) {
      
     }
    }
    

    相关文章

      网友评论

      • 时彬斌:写的最好的,没有之一,我之前用了@valid注解,也是只报500错误,没有具体的错误信息返回,若要返回信息,需要做一个aop层统一把异常处理掉,将错误信息放进Errors或是集成了Errors的结果集中例如BindingResult
        joyhj: @时彬斌 谢谢鼓励,一起加油😊
      • edc81fe714ab:我见过写的最好的
        joyhj:谢谢
      • weixk:还有个问题请教一下,当我使用方法级别的校验时,像这种:
        @NotEmpty(message = "昵称不能为空") @RequestParam(value = "nickname") String nickname
        但是异常没有将message的信息带出
        {
        "timestamp": 1481218866065,
        "status": 500,
        "error": "Internal Server Error",
        "exception": "javax.validation.ConstraintViolationException",
        "message": "No message available",
        "path": "/user/register"
        }
        有没有遇到过类似问题?
        weixk:@vasthua 谢谢,已经决定把参数校验放到类里面了。SpringBoot官方文档里也没有提到在方法参数上校验,但是看别人的博客提到过。应该是我工程配置或是使用版本有些不一样的原因吧,我放在类里校验
        @RequestMapping(value = "/login", method = RequestMethod.POST)
        public Result<User> login(@Valid UserLogin userLogin)
        抛出的异常是org.springframework.validation.BindException
        而网上教程一般是说org.springframework.web.bind.MethodArgumentNotValidException
        joyhj:@weixk 按照你给出的代码进行分析,进行测试,应该会出现org.springframework.web.bind.MissingServletRequestParameterException这个异常,可以捕获这个异常添加返回值。不知道你给出的异常信息为什么会出现,可以给出更详细的代码进行进一步分析。
      • weixk:请问,SpringBoot验证失败返回的默认信息怎么修改?不需要详细的出错信息,就是变成这种格式{code:0,message:""}
        weixk:@vasthua thx
        joyhj:@weixk 验证失败会抛出MethodArgumentNotValidException 异常,可以捕获这个异常,然后增加返回值。具体的写法可以搜索@ControllerAdvice, @ExceptionHandler这两个注解看看

      本文标题:Spring boot使用总结(三)校验

      本文链接:https://www.haomeiwen.com/subject/wbfwsttx.html