最重要的SpringMVC注释之一是@ModelAttribute注释。
@ModelAttribute是一个批注,它将方法参数或方法返回值绑定到命名模型属性,然后将其公开给 Web 视图。
在本教程中,我们将通过一个通用概念(公司员工提交的表单)来演示此注释的可用性和功能。
了解Spring MVC提供的接口Model,ModelMap和ModelAndView。
阅读更多→
春季@RequestParam注释的详细指南
阅读更多→
正如介绍性段落所揭示的,我们可以将其用作方法参数@ModelAttribute也可以在方法级别使用。
当我们在方法级别使用注释时,它表明该方法的目的是添加一个或多个模型属性。此类方法支持与@RequestMapping方法相同的参数类型,但它们不能直接映射到请求。
让我们看一个快速示例来了解它是如何工作的:
- @ModelAttribute
- public void addAttributes(Model model) {
- model.addAttribute("msg", "Welcome to the Netherlands!");
- }
在上面的示例中,我们看到一个方法,该方法将一个名为msg的属性添加到控制器类中定义的所有模型s。
当然,我们将在本文后面看到这一点。
通常,Spring MVC 总是在调用任何请求处理程序方法之前先调用该方法。基本上,@ModelAttribute方法在调用注释的控制器方法之前@RequestMapping被调用。这是因为必须在控制器方法内部开始任何处理之前创建模型对象。
同样重要的是,我们将相应的类注释为@ControllerAdvice。因此,我们可以在模型中添加将被标识为全局的值。这实际上意味着对于每个请求,响应中的每个方法都存在默认值。
当我们使用注释作为方法参数时,它指示从模型中检索参数。当注释不存在时,应首先实例化它,然后将其添加到模型中。一旦出现在模型中,参数字段应从具有匹配名称的所有请求参数中填充。
在下面的代码片段中,我们将使用提交到addEmployee终结点的表单中的数据填充员工模型属性。Spring MVC 在调用提交方法之前在幕后执行此操作:
- @RequestMapping(value = "/addEmployee", method = RequestMethod.POST)
- public String submit(@ModelAttribute("employee") Employee employee) {
- // Code that uses the employee object
-
- return "employeeView";
- }
在本文的后面部分,我们将看到如何使用employee对象填充employeeView模板的完整示例。
它将表单数据与 Bean 绑定。批注的控制器@RequestMapping可以具有批注的自定义类参数@ModelAttribute。
在Spring MVC中,我们将其称为数据绑定,这是一种通用机制,使我们不必单独解析每个表单字段。
在本节中,我们将查看概述部分中概述的示例,这是一个非常基本的表单,提示用户(特别是公司员工)输入一些个人信息(特别是姓名和ID)。提交完成后,并且没有任何错误,用户希望在另一个屏幕上看到以前提交的数据。
让我们首先创建一个包含 id 和名称字段的简单表单:
- <form:form method="POST" action="/spring-mvc-basics/addEmployee"
- modelAttribute="employee">
- <form:label path="name">Name</form:label>
- <form:input path="name" />
-
- <form:label path="id">Id</form:label>
- <form:input path="id" />
-
- <input type="submit" value="Submit" />
- </form:form>
下面是控制器类,我们将在其中实现上述视图的逻辑:
- @Controller
- @ControllerAdvice
- public class EmployeeController {
-
- private Map<Long, Employee> employeeMap = new HashMap<>();
-
- @RequestMapping(value = "/addEmployee", method = RequestMethod.POST)
- public String submit(
- @ModelAttribute("employee") Employee employee,
- BindingResult result, ModelMap model) {
- if (result.hasErrors()) {
- return "error";
- }
- model.addAttribute("name", employee.getName());
- model.addAttribute("id", employee.getId());
-
- employeeMap.put(employee.getId(), employee);
-
- return "employeeView";
- }
-
- @ModelAttribute
- public void addAttributes(Model model) {
- model.addAttribute("msg", "Welcome to the Netherlands!");
- }
- }
在submit() 方法中,我们有一个绑定到视图的Employee对象。我们可以简单地将表单字段映射到对象模型。在该方法中,我们从窗体中获取值并将其设置为模型映射。
最后,我们返回employeeView,这意味着我们将相应的 JSP 文件作为View代表调用。
此外,还有一个addAttributes()方法。其目的是在模型中添加将全局标识的值。也就是说,对每个控制器方法的每个请求都将返回默认值作为响应。我们还必须将特定类注释为@ControllerAdvice。
如前所述,Model对象非常简单,包含“前端”属性所需的所有内容。现在让我们看一个例子:
- @XmlRootElement
- public class Employee {
-
- private long id;
- private String name;
-
- public Employee(long id, String name) {
- this.id = id;
- this.name = name;
- }
-
- // standard getters and setters removed
- }
@ControllerAdvice协助控制器,特别是@ModelAttribute适用于所有方法的方法@RequestMapping。当然,我们的addAttributes() 方法将首先运行,先于其他@RequestMapping方法。
记住这一点,在submit() 和addAttributes() 都运行之后,我们可以在从 Controller类返回的视图中引用它们,方法是在美元化的花括号二重奏中提及它们的名字,例如 ${name}。
现在让我们打印我们从表单中收到的内容:
- <h3>${msg}</h3>
- Name : ${name}
- ID : ${id}
在本文中,我们研究了方法参数和方法级用例中@ModelAttribute注释的使用。
本文的实现可以在github项目中找到。