@ModelAttribute和抽象类

时间:2016-05-18 13:07:51

标签: spring spring-mvc inheritance abstract-class

我知道有类似的问题。他们给出的例子太过零碎而且不清楚。

我需要通过发送POST的页面上的表单来编辑实体。标准方法是控制器中的方法使用带有@ModelAttribute和validator的参数。如果一个表单服务于抽象类的某个子类,则生成必要字段没有问题,但控制器中存在问题。

据我了解,@ ModelAttribute以这种方式工作:它初始化所需的对象类,然后收集他的请求参数的字段。当然,如果对象是抽象类,则无法初始化它。因此,表单有一个字段,指示要初始化的子类。接下来,我们需要代码的和平,它将读取此属性并初始化正确的子类。它应该是什么?我看到了关于Converter,PrepertyEditor,WebDataBinder的零碎例子,但很难将所有内容放在一起。

因此。有以下层次结构:

public abstract class Person {role, name, email, password ...}
public class Student extends Person {}
public class Lecturer extends Person {}

其中有一个控制器和方法:

@RequestMapping (Path = "/ persons / uid {personId} / edit",
                method = RequestMethod.GET)
public String editPerson (@PathVariable Integer personId, Model model) {
    Person find = personDAO.read (personId);
    model.addAttribute ( "person", find);
    return "editPerson";
}

@RequestMapping (Path = "/ persons / uid {personId} / edit",
                method = RequestMethod.POST)
public String editPersonPost (@PathVariable Integer personId,
        @Valid @ModelAttribute ( "Person") Person person,
        BindingResult result) {
    if (result.hasErrors ()) return "editPerson error = true?";
    personDAO.update (person);
    return "redirect: / persons / uid" + personId + "saved = true?";
}

还有一个带有以下形式的JSP:

<%@ page language="java" contentType="text/html; charset=UTF-8"
    pageEncoding="UTF-8"%>

<%@taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<%@taglib uri="http://www.springframework.org/tags/form" prefix="form"%>

<h1>${person.name}</h1>

<form:form action="edit" method="post" commandName="person">
    <input type="hidden" value="${person.role}" name="person_type" />
    <table>
        <tr>
            <td>Password</td>
            <td><form:input path="httpAuth.password" type="password"/></td>
            <td><form:errors path="httpAuth.password" cssClass="error"></form:errors></td>
        </tr>
        <tr>
            <td>Email</td>
            <td><form:input path="email" /></td>
            <td><form:errors path="email" cssClass="error"></form:errors></td>
        </tr>
        <tr>
            <td></td>
            <td><input type="submit" value="Save"></td>
            <td></td>
        </tr>
    </table>
</form:form>

此外,转换器已编写,但我怀疑是否有必要,或者做它(继承另一个类...)

public class PersonConverter implements Converter <String, Person> {

public Person convert (String personType) {
    Person person = null;
    switch (personType) {
        case "Student":
            person = new Student ();
            break;
        case "Lecturer":
            person = new Lecturer ();
            break;
        default:
            throw new IllegalArgumentException (
                    "Unknown person type:" + personType);
    }
    return person;
}}

在ConversionService中注册了

<bean class="org.springframework.context.support.ConversionServiceFactoryBean"
    id="theConversionService">
    <property name="converters">
        <list>
            <bean class="schedule.service.PersonConverter"></bean>
        </list>
    </property>
</bean>
<mvc:annotation-driven conversion-service="theConversionService" validator="validator"/>

然而,缺少某些东西,这是一种从请求参数中获取person_type并将其提供给转换器的方法,它将通过自动绑定机制返回控制器方法的结果。

请帮帮我。

1 个答案:

答案 0 :(得分:1)

您只需要确保下面的元素

<input type="hidden" value="${person.role}" name="person_type" />

将其命名属性更改为人

<input type="hidden" value="${person.role}" name="person" />

以便它与控制器中的model属性匹配

public String editPersonPost (@PathVariable Integer personId,
        @Valid @ModelAttribute ( "person") Person person,
        BindingResult result)

这是它的工作原理。

当收到请求并且Spring需要创建模型属性时,它会检查该属性是否已存在。如果它不存在并且没有匹配名称的请求参数,则使用参数类的默认构造函数

创建一个新对象

如果它存在并且与参数类型匹配,则它继续绑定请求参数。如果它不兼容或者同名的请求参数可用,它会尝试找到能够将当前值转换为所需类型的转换器

如果转换成功,它会将请求参数绑定到结果,否则会抛出异常

在您的情况下,person属性作为String发送。 Spring会尝试将其转换为Person。它选择你的PersonConverter在绑定之前转换到适当的子类