文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

如何使用@Valid+BindingResult进行controller参数校验

2023-06-21 22:01

关注

这篇文章主要介绍“如何使用@Valid+BindingResult进行controller参数校验”,在日常操作中,相信很多人在如何使用@Valid+BindingResult进行controller参数校验问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”如何使用@Valid+BindingResult进行controller参数校验”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

@Valid+BindingResult进行controller参数校验

由于controller是调用的第一层,经常参数校验将在这里完成,常见有非空校验、类型校验等,常见写法为以下伪代码:

public void round(Object a){  if(a.getLogin() == null){     return "手机号不能为空!";   }}

但是调用对象的位置会有很多,而且手机号都不能为空,那么我们会想到把校验方法抽出来,避免重复的代码。但有框架支持我们通过注解的方式进行参数校验。

先立个场景,为往动物园添加动物,动物对象如下,时间节点大概在3030年,我们认为动物可登陆动物专用的系统,所以有password即自己的登录密码。

public class Animal {    private String name;    private Integer age;    private String password;    private Date birthDay;}

调用创建动物的controller层如下,简洁明了,打印下信息后直接返回。

@RestController@RequestMapping("/animal")public class AnimalController {   @PostMapping    public Animal createAnimal(@RequestBody Animal animal){        logger.info(animal.toString());        return animal;    }}

伪造Mvc调用的测试类。

@RunWith(SpringRunner.class)@SpringBootTestpublic class TestAnimal {     private final static Logger logger = LoggerFactory.getLogger(TestAnimal.class);    @Autowired    private WebApplicationContext wac;    private MockMvc mockMvc;     @Before    public void initMock(){        mockMvc = MockMvcBuilders.webAppContextSetup(wac).build();    }     @Test    public void createAnimal() throws Exception {        String content = "{\"name\":\"elephant\",\"password\":null,\"birthDay\":"+System.currentTimeMillis()+"}";        String result = mockMvc.perform(MockMvcRequestBuilders.post("/animal")                .content(content)                .contentType(MediaType.APPLICATION_JSON_UTF8))                .andExpect(MockMvcResultMatchers.status().isOk())                .andReturn().getResponse().getContentAsString();        logger.info(result);    }}

以上代码基于搭建的springboot项目,想搭建的同学可以参考搭建篇 https://www.yisu.com/article/226998.htm

代码分析,日期格式的参数建议使用时间戳传递,以上birthDay传递 "2018-05-08 20:00:00",将会抛出日期转换异常,感兴趣的同学可以试试。

由于密码很重要,现在要求密码为必填,操作如下,添加@NotBlank注解到password上:

@NotBlankprivate String password;

但光加校验注解是不起作用的,还需要在方法参数上添加@Valid注解,如下:

@Valid @RequestBody Animal animal

此时执行测试方法,抛出异常,返回状态为400:

java.lang.AssertionError: Status
Expected :200
Actual :400
<Click to see difference>


at org.springframework.test.util.AssertionErrors.fail(AssertionErrors.java:54)
at org.springframework.test.util.AssertionErrors.assertEquals(AssertionErrors.java:81)

说明对password的非空校验已经生效了,直接抛出异常。如果不想抛出异常,想返回校验信息给前端,这个时候就需要用到BindingResult了,修改创建动物的方法,添加BindingResult参数:

@PostMapping    public Animal createAnimal(@Valid @RequestBody Animal animal, BindingResult bindingResult){        if (bindingResult.hasErrors()){            bindingResult.getAllErrors().forEach(o ->{                FieldError error = (FieldError) o;                logger.info(error.getField() + ":" + error.getDefaultMessage());            });        }        logger.info(animal.toString());        return animal;    }

此时,执行测试,可以看到日志中的错误信息:

2018-05-09 00:59:37.453 INFO 14044 --- [ main] c.i.s.d.web.controller.AnimalController : password:may not be empty

为了满足我们编码需要我们需要进行代码改造,1.不能直接返回animal。2.返回的提示信息得是用户可读懂的信息。

controller方法改造如下,通过Map对象传递请求成功后的信息或错误提示信息。

@PostMapping    public Map<String,Object> createAnimal(@Valid @RequestBody Animal animal, BindingResult bindingResult){        logger.info(animal.toString());        Map<String,Object> result = new HashMap<>();        if (bindingResult.hasErrors()){            FieldError error = (FieldError) bindingResult.getAllErrors().get(0);            result.put("code","400");//错误编码400            result.put("message",error.getDefaultMessage());//错误信息            return result;        }        result.put("code","200");//成功编码200        result.put("data",animal);//成功返回数据        return result;    }

返回的密码提示信息如下:

@NotBlank(message = "密码不能为空!")private String password;

执行测试方法,返回结果

com.imooc.security.demo.TestAnimal : {"code":"400","message":"密码不能为空!"}

最后贴一个,设置password值返回成功的信息

com.imooc.security.demo.TestAnimal : {"code":"200","data":{"name":"elephant","age":null,"password":"lalaland","birthDay":1525799768955}}

由于篇幅有限,下次会以这个实例为基础,实现一个自定义的注解实现,该篇文章到此结束。

Controller层方法的参数校验

import com.example.demo.pojo.Student;import org.springframework.stereotype.Controller;import org.springframework.web.bind.annotation.PostMapping;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.ResponseBody; import javax.validation.Valid; @Controller@RequestMapping("/stu")public class StudentController {     @PostMapping("/addStu")    @ResponseBody    public String addStudent(@Valid Student student){        System.out.println("存储student对象");        System.out.println(student);        return "ok";    } }
import org.hibernate.validator.constraints.Length; import javax.validation.constraints.Max;import javax.validation.constraints.Min;import javax.validation.constraints.NotEmpty;import javax.validation.constraints.NotNull; public class Student {     @NotNull(message = "传入的是空值,请传值")    @Min(value = 0,message = "传入学生分数有误,分数在0-100之间")    @Max(value = 100,message = "传入学生分数有误,分数在0-100之间")    private Integer score;     @NotEmpty(message = "传入的是空字符串,请传值")    @NotNull(message = "传入的是空值,请传值")    private String name;     @NotNull(message = "传入的是空值,请传值")    @NotEmpty(message = "传入的是空字符串,请传值")    @Length(min = 11,max = 11,message = "号码有误,长度应为11位")    private String mobile;     public Integer getScore() {        return score;    }     public void setScore(Integer score) {        this.score = score;    }     public String getName() {        return name;    }     public void setName(String name) {        this.name = name;    }     public String getMobile() {        return mobile;    }     public void setMobile(String mobile) {        this.mobile = mobile;    }     @Override    public String toString() {        return "Student{" +                "score=" + score +                ", name='" + name + '\'' +                ", mobile='" + mobile + '\'' +                '}';    }}

全局统一异常拦截器

import org.springframework.validation.BindException;import org.springframework.web.bind.annotation.ControllerAdvice;import org.springframework.web.bind.annotation.ExceptionHandler;import org.springframework.web.bind.annotation.ResponseBody;  @ControllerAdvicepublic class GlobalExceptionInterceptor {     @ExceptionHandler(value = Exception.class)    @ResponseBody    public String exceptionHandler(Exception e){        String failMessage=null;         if(e instanceof BindException){            failMessage=((BindException) e).getBindingResult().getFieldError().getDefaultMessage();        }        return failMessage;    } }

如何使用@Valid+BindingResult进行controller参数校验

如何使用@Valid+BindingResult进行controller参数校验

当我们传入的参数有误时,就会被异常拦截器捕获,返回给我们错误信息。

如何使用@Valid+BindingResult进行controller参数校验

到此,关于“如何使用@Valid+BindingResult进行controller参数校验”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     801人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     348人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     311人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     432人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     220人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯