3.十、在方法参数上使用@ModelAttribute注解

  这一部分实例见这个项目的 mvc 分支下的 ModelAttrArgsController.javahtml


  上节说过,@ModelAttribute能够用在方法或方法参数上。本节解释它用在参数上的情形。java

  一个用在方法参数上的@ModelAttribute注解指示了参数应该从模型(这里所说的“模型”指 Model)中获取。若是模型中不存在,参数会首先被实例化,而后添加到模型中。一旦模型中存在,这个参数的字段会被全部的名字匹配的请求参数所填充。这在 Spring MVC 中叫作数据绑定,它可以把你从要对每个字段进行类型转换的繁重体力劳动中解救出来,是很是有用的机制。git

@PostMapping("/owners/{ownerId}/pets/{petId}/edit")
public String processSubmit(@ModelAttribute Pet pet) 
{ 
    // ...
}

  上面给出的例子的 Pet 实例来自哪里?这里有一个可选项:web

  • 它可能已经存在与模型中了,由于使用了@SessionAttributes — 见“使用@SessionAttributes存储模型属性到 HTTP 会话中”一节.spring

  • 它可能已经存在于模型中了,由于同一个控制器中的@ModelAttribute方法,就像上一节中解释的那样。数据库

  • 它多是从URI模板变量和类型转换器中获取的(下面会详细解释)。session

  • 它多是使用默认构造器初始化的。mvc

@ModelAttribute方法是一种经常使用的从数据库中获取属性的方式,能够经过使用@SessionAttributes注解把这种属性在各个请求之间共享。在一些状况下,能够很方便的经过使用URI模板变量和类型转换器来获取这些属性。下面是一个例子:app

@PutMapping("/accounts/{account}")
public String save(@ModelAttribute("account") Account account) 
{
    // ...
}

  在这个例子中,模型属性名(“account”)匹配URI模板变量名。若是你注册的一个Converter<String, Account>能够把字符串 account 值转换为一个 Account,那么上面的例子即便不须要@ModelAttribute也能够正常工做。.net

  下一步是数据绑定。WebDataBinder类匹配请求参数名称——包括请求字符串参数和表单字段——到属性字段名。在必须的类型转换(从字符串到目标类型字段)以后,匹配的字段收集好了。数据绑定和校验见官方文档第五章-检验、数据绑定和类型转换。为控制器自定义数据绑定过程见 “自定义WebDataBinder初始化”一节。

  在数据绑定以后,可能会出现一些错误,好比缺乏必须字段或者类型转换错误。为了检查这些错误,你能够紧跟在@ModelAttribute参数后面添加一个BindingResult参数:

@PostMapping("/owners/{ownerId}/pets/{petId}/edit")
public String processSubmit(@ModelAttribute("pet") Pet pet, BindingResult result) 
{
    if (result.hasErrors()) { return "petForm"; }
    // ...
}

  有了BindingResult以后,你就能够检查在提交同一个表单时有没有发现错误,这些错误能够经过 Spring 的<errors>表单标签来显示。

  注意到在一些状况下,在 model 中不使用数据绑定来访问一个属性是颇有用的。这种状况下,你能够把Model注入到控制器中,或者在注解上使用绑定标记:

@ModelAttribute
public AccountForm setUpForm() { return new AccountForm(); }

@ModelAttribute
public Account findAccount(@PathVariable String accountId) 
{ 
    return accountRepository.findOne(accountId); 
}

@PostMapping("update")
public String update(
        @Valid AccountUpdateForm form, BindingResult result,
        @ModelAttribute(binding=false) Account account) 
{ /* omitted */ }

  除了数据绑定,你也可使用你本身定义的、用来传递BindingResult(用于记录数据绑定错误)的校验器调用校验这容许数据绑定和验证的错误积累在一个地方,随后报告给用户:

@PostMapping("/owners/{ownerId}/pets/{petId}/edit")
public String processSubmit(@ModelAttribute("pet") Pet pet, BindingResult result) 
{
    new PetValidator().validate(pet, result);
    if (result.hasErrors()) return "petForm";

    // ...
}

  或者你能够自动调用校验,经过添加 JSR-303 的 @Valid 注解:

@PostMapping("/owners/{ownerId}/pets/{petId}/edit")
public String processSubmit(@Valid @ModelAttribute("pet") Pet pet, BindingResult result) 
{
    if (result.hasErrors()) return "petForm";

    // ...
}

  关于如何配置和使用校验器,详见官方文档的5.8节“Spring校验”和第五章 校验、数据绑定和类型转换

相关文章
相关标签/搜索