spring-mvc 中抽象类的数据绑定

2022-09-04 04:49:16

我已经浏览了Spring文档和源代码,但仍然没有找到我问题的答案。

我的领域模型中有这些类,并希望将它们用作spring-mvc中的支持表单对象。


public abstract class Credentials {
  private Long     id;
  ....
}
public class UserPasswordCredentials extends Credentials {
  private String            username;
  private String            password;
  ....
}
public class UserAccount {
  private Long              id;
  private String            name;
  private Credentials       credentials;
  ....
}

我的控制器:


@Controller
public class UserAccountController
{
  @RequestMapping(value = "/saveAccount", method = RequestMethod.POST)
  public @ResponseBody Long saveAccount(@Valid UserAccount account)
  {
    //persist in DB
    return account.id;
  }

  @RequestMapping(value = "/listAccounts", method = RequestMethod.GET)
  public String listAccounts()
  {
    //get all accounts from DB
    return "views/list_accounts";
  }
  ....
}

在UI上,我有不同凭据类型的动态表单。我的 POST 请求通常如下所示:


name                    name
credentials_type        user_name
credentials.password    password
credentials.username    username

如果我尝试向服务器提交请求,则会引发以下异常:


org.springframework.beans.NullValueInNestedPathException: Invalid property 'credentials' of bean class [*.*.domain.UserAccount]: Could not instantiate property type [*.*.domain.Credentials] to auto-grow nested property path: java.lang.InstantiationException
    org.springframework.beans.BeanWrapperImpl.newValue(BeanWrapperImpl.java:628)

我最初的想法是使用@ModelAttribute


    @ModelAttribute
    public PublisherAccount prepareUserAccountBean(@RequestParam("credentials_type") String credentialsType){
      UserAccount userAccount = new PublisherAccount();
      Class credClass = //figure out correct credentials class;
      userAccount.setCredentials(BeanUtils.instantiate(credClass));
      return userAccount;
    }

这种方法的问题在于,在任何其他方法(如)之前调用方法也是不合适的。prepareUserAccountBeanlistAccounts

一个强大的解决方案是将两者移出并移动到单独的控制器。这听起来不对:我希望所有与用户相关的操作都驻留在同一个控制器类中。prepareUserAccountBeansaveUserAccount

任何简单的解决方案?我可以以某种方式使用DataBinder,PropertyEditor或WebArgumentResolver吗?

谢谢!!!!!


答案 1

我看不到任何简单而优雅的解决方案。也许因为问题不在于如何在Spring MVC中数据绑定抽象类,而是:为什么首先在形式对象中使用抽象类?我认为你不应该。

从表单发送到控制器的对象被称为“表单(支持)对象”,这是有原因的:对象属性应该反映表单字段。如果您的表单具有用户名和密码字段,则您的类中应具有用户名和密码属性。

所以应该有一个类型。这将跳过“抽象实例化尝试”错误。为此有两种解决方案:credentialsUserPasswordCredentials

  • 建议:将 UserAccount.credentials 的类型从 Credentials 更改为 UserPasswordCredentials。我的意思是,除了UserPasswordCredentials之外,UserAccount可能具有哪些凭据?更重要的是,我敢打赌,您的数据库用户帐户具有作为凭据存储的用户名和密码,因此您也可以直接在UserAccount中键入UserPasswordCredentials。最后,Spring建议使用“现有的业务对象作为命令或表单对象”(参见文档),因此修改UserAccount将是要走的路。
  • 不推荐 :保持 UserAccount 不变,并创建一个 UserAccountForm 类。此类将具有与 UserAccount 相同的属性,不同之处在于 UserAccountForm.credentials 具有 UserPasswordCredentials 类型。然后,在列出/保存时,一个类(例如UserAccountService)进行转换。此解决方案涉及一些代码重复,因此只有在有充分理由(无法更改的旧实体等)时才使用它。

答案 2

我不确定,但您应该在控制器上使用ViewModel类,而不是域对象。然后,在 saveAccount 方法中,您将验证此 ViewModel,如果一切正常,则将其映射到域模型中并保留它。

通过这样做,您还有另一个优势。如果您将任何其他属性添加到您的域 UserAccount 类,例如:private bool isAdmin。如果您的 Web 用户向您发送了一个 isAdmin=true 的 POST 参数,则该参数将绑定到用户域类并持久保存。

好吧,这就是我的做法:

public class NewUserAccount {
    private String name;
    private String username;
    private String password;
}  

@RequestMapping(value = "/saveAccount", method = RequestMethod.POST)
public @ResponseBody Long saveAccount(@Valid NewUserAccount account)
{
    //...
}

推荐