一尘不染

Spring Web MVC-验证单个请求参数

java

我在Spring Web MVC 3.0中运行一个webapp,我有许多控制器方法,其签名大致如下:

@RequestMapping(value = "/{level1}/{level2}/foo", method = RequestMethod.POST)
public ModelAndView createFoo(@PathVariable long level1,
        @PathVariable long level2,
        @RequestParam("foo_name") String fooname,
        @RequestParam(value = "description", required = false) String description);

我想添加一些验证-
例如,description应限制为一定长度或fooname仅包含某些字符。如果此验证失败,则我想向用户返回一条消息,而不仅仅是抛出一些未经检查的异常(如果我让数据渗透到DAO层,无论如何都会发生这种情况)。我知道JSR303,但尚未使用它,并且不太了解如何在Spring上下文中应用它。

据我了解,另一种选择是将绑定@RequestBody到整个域对象并在其中添加验证约束,但是目前,我的代码已设置为接受单个参数,如上所示。

使用这种方法将验证应用于输入参数的最直接方法是什么?


阅读 250

收藏
2020-09-08

共1个答案

一尘不染

没有什么内置做到这一点,不是
尚未反正。在当前发行版中,如果要进行自动验证,则仍然需要使用WebDataBinder将参数绑定到对象上。如果您使用的是SpringMVC,则值得学习,即使这不是您执行此任务的首选。

看起来像这样:

public ModelAndView createFoo(@PathVariable long level1,
        @PathVariable long level2,
        @Valid @ModelAttribute() FooWrapper fooWrapper,
        BindingResult errors) {
  if (errors.hasErrors() {
     //handle errors, can just return if using Spring form:error tags.
  }
}

public static class FooWrapper {
  @NotNull
  @Size(max=32)
  private String fooName;
  private String description;
//getset
}

如果您在类路径上具有Hibernate Validator 4或更高版本,并使用默认的分派器设置,则它应该“正常工作”。

由于评论变得越来越大,因此进行编辑:

这是在这不是“预期”的人Spring知道如何注入,如一个你的方法签名的任何对象HttpRequestModelMap等等,将得到的数据绑定。对于简单情况,只需将请求参数名称与bean属性名称进行匹配并调用setter即可实现。该@ModelAttribute存在的仅仅是个人风格的东西,在这种情况下,它没有做任何事情。JSR-303与@Valid在方法参数上的集成通过插入WebDataBinder。如果您使用@RequestBody,则使用的是基于spring确定请求正文的内容类型的对象编组器(通常仅来自http标头。)分派器servlet(AnnotationMethodHandlerAdapter确实)没有办法为任何封送编组“翻转验证开关”。它只是将Web请求的内容传递到消息转换器,并返回一个Object。不会生成BindingResult对象,因此无论如何都无法设置错误。

您仍然可以将验证器注入控制器中,然后在获取的对象上运行它,它只是与@Validon参数中的on请求参数没有魔术集成BindingResult

2020-09-08