3

不是重复的,请参阅附加说明

我想绑定如下模型设置

public class Shop{
    public string Name {get;set;}
    public ICollection<Product> Products {get;set;} //Product is abstract
}

public abstract class Product{
    public string Name {get;set;}
}

public class ProductA : Product{
    public string foo {get;set;}
}

public class ProductB :Product{
    public string bar {get;set;}
}

像这样的控制器

public ActionResult(){
    Shop model = ShopFactory.GetShop();
    return View(model);
}

[HttpPost]
public ActionResult(Shop model){
    //....
}

我正在使用BeginCollectionItem绑定集合,但是在发布表单时出现问题,因为它无法创建抽象类 - 即 Shop.Products 中的对象

我已经查看了DefaultModelBinder要覆盖的子类,CreateModel但是 CreateModel 永远不会使用参数调用modeltype = Product,只有modeltype = Shop

如何创建将绑定具有抽象集合作为属性的对象的 ModelBinder?

澄清
这个问题不是重复的,因为我们不是在处理抽象模型,我们正在处理一个具有抽象对象集合的模型,这在模型绑定系统中经历了一个单独的过程。

4

3 回答 3

3

回答说,

我遇到的问题是我正在创建一个 ModelBinder Shop,而不是Products.

简单的。

更新
由于这被否决了,我想我会澄清一下。

我试图模态绑定Shop类,因为那是我发送到视图的类。在我看来这是有道理的,因为那是我绑定的模型。问题是 DefaultModelBinder 中的一个方法在遇到IEnumerable. 所以解决方案是继承 DefaultModelBinder

public class ProductModelBinder : DefaultModelBinder{
    protected override object CreateModel(ControllerContext controllerContext, ModelBindingContext bindingContext, Type modelType)
    {
        if (modelType.Equals(typeof(Product)))
        {
            //For now only support Product1's
            // Todo: Add support for different types
            Type instantiationType = typeof(Product1);
            var obj = Activator.CreateInstance(instantiationType);
            bindingContext.ModelMetadata = ModelMetadataProviders.Current.GetMetadataForType(null, instantiationType);
            bindingContext.ModelMetadata.Model = obj;
            return obj;
        }

        return base.CreateModel(controllerContext, bindingContext, modelType);
    }

并将所述子类注册到活页夹:

ModelBinders.Binders.Add(new KeyValuePair<Type, IModelBinder>(typeof(FormItem), new Forms.Mvc.FormItemModelBinder()));
于 2013-07-24T02:11:54.013 回答
3

澄清一下,对于像我这样偶然发现此页面并寻找解决方案的其他人:

绑定子类型(抽象类或接口)的 MVC 3 模型正是您要寻找的。由于 BeginCollectionItem,您不需要做任何特别的事情。

只需为您拥有的任何抽象类编写一个模型绑定器,并使用以下内容装饰抽象类:

[ModelBinder(typeof(MyModelBinder))]
public abstract class MyClass { ...

然后在您的 ModelBinder 中,使用如下内容:

string typeName = bindingContext.ValueProvider.GetValue(bindingContext.ModelName + ".type").AttemptedValue;
Type instantiationType = Type.GetType(typeName);

从 EditorTemplate 的隐藏字段中获取类型:

    @Html.HiddenFor(m => type)

最后,如果您仍然无法获得具体类型的属性,请仔细检查它们是否实际上是属性,而不是字段!

于 2014-11-28T11:52:36.130 回答
0

我认为您应该编写自定义模型 binder。请参阅如何将抽象类或接口的集合绑定到模型 - ASP.NET MVCMVC 3 模型绑定子类型(抽象类或接口)

public class MyModelBinder : DefaultModelBinder
{
    protected override object CreateModel(ControllerContext controllerContext, ModelBindingContext bindingContext, Type modelType)
    {
        /// MyBaseClass and MyDerievedClass are hardcoded.
        /// We can use reflection to read the assembly and get concrete types of any base type
        if (modelType.Equals(typeof(MyBaseClass)))
        {
            Type instantiationType = typeof(MyDerievedClass);                
            var obj=Activator.CreateInstance(instantiationType);
            bindingContext.ModelMetadata = ModelMetadataProviders.Current.GetMetadataForType(null, instantiationType);
            bindingContext.ModelMetadata.Model = obj;
            return obj;
        }
        return base.CreateModel(controllerContext, bindingContext, modelType);
    }

}
于 2013-07-22T06:31:59.257 回答