public class MyViewModel { public BaseViewModel mySubViewModel; } public class ChildViewModel: BaseViewModel {}
然后我创建一个MyViewModel模型,其中包含ChildViewModel类型的属性.在视图中,它显示得很好.
然后我点击保存按钮向模型提交更改,然后调用以下控制器:
[HttpPost] public ActionResult Edit(MyViewModel model) { return null; }
令我惊讶的是,属性mySubViewModel现在是BaseViewModel类型而不是ChildViewModel!我不知道这里发生了什么.我究竟做错了什么 ?
To my surprise, the property mySubViewModel is now of type
BaseViewModel instead of ChildViewModel ! I have no idea what’s going
on here. What am I doing wrong ?
你不应该对此感到惊讶.你没有做错任何事.这是设计的.默认模型绑定器看到您的控制器操作正在采用MyViewModel参数并尝试将POST请求的内容绑定到它.默认的模型绑定器绝对无法知道您可能已经编写了一些派生类(例如,在您的情况下为ChildViewModel),并且您希望在此处实例化这些派生类.您已将此具体视图模型实例从GET操作传递到视图的事实对POST操作没有影响.根据HTTP和默认模型绑定器看到的内容来考虑它.例如,可以从一些完全不同的客户端调用此POST操作,而不是从表单提交调用.例如,可以是对POST操作执行HTTP请求的iPhone应用程序.看,现在它非常有意义.默认模型绑定器只能看到POST请求的有效负载和您指定为操作参数的类型.这就是他所做的=>它实例化此类型并从POST有效内容数据绑定其属性.
因此,一种可能性是编写一个自定义模型绑定器,它将实例化您希望的视图模型的具体实例.我在this post
举例说明了这样一个自定义模型绑定器.在这个例子中,我在表单中包含了一个隐藏字段,它将包含我们想要实例化的视图模型的具体类型,然后自定义模型绑定器只是使用这些信息在运行时创建此类型.