c# - ASP.NET MVC 3 : DefaultModelBinder with inheritance/polymorphism

标签 c# inheritance asp.net-mvc-3 mvccontrib ninject-2

首先,对于大型文章(我首先尝试做一些研究)以及针对同一问题的技术混合(ASP.NET MVC 3,Ninject和MvcContrib)感到抱歉。

我正在使用ASP.NET MVC 3开发一个项目来处理一些客户订单。

简而言之:我有一些对象是继承自抽象类Order的对象,当对 Controller 发出POST请求时,我需要解析它们。如何解析正确的类型?我是否需要重写DefaultModelBinder类,或者有其他方法可以做到这一点?有人可以为我提供一些代码或其他链接来执行此操作吗?任何帮助将是巨大的!
如果帖子令人困惑,我可以进行任何更改以使其清楚!

因此,对于需要处理的订单,我具有以下继承树:

public abstract partial class Order {

    public Int32 OrderTypeId {get; set; }

    /* rest of the implementation ommited */
}

public class OrderBottling : Order { /* implementation ommited */ }

public class OrderFinishing : Order { /* implementation ommited */ }

这些类都是由Entity Framework生成的,因此我将不修改它们,因为我需要更新模型(我知道可以扩展它们)。同样,会有更多的订单,但是所有订单都来自Order

我有一个通用 View (Create.aspx)来创建订单,此 View 为每个继承的订单(在本例中为OrderBottlingOrderFinishing)调用一个强类型的局部 View 。我在Create()类上为GET请求和POST请求定义了OrderController方法。第二个如下:
public class OrderController : Controller
{
    /* rest of the implementation ommited */

    [HttpPost]
    public ActionResult Create(Order order) { /* implementation ommited */ }
}

现在的问题是:当我从表单接收到带有数据的POST请求时,MVC的默认绑定(bind)程序尝试实例化Order对象,这是可以的,因为方法的类型就是那个。但是因为Order是抽象的,所以它不能被实例化,这是应该做的。

问题:,如何发现 View 发送的是哪种具体的Order类型?

我已经在Stack Overflow上进行过搜索,并在Google上进行了很多搜索(我正在研究此问题大约3天了!),并找到了一些解决类似问题的方法,但是我找不到真正的东西问题。解决此问题的两种选择:
  • 覆盖ASP.NET MVC DefaultModelBinder并使用直接注入(inject)来发现Order是哪种类型;
  • 为每个订单创建一个方法(不美观,维护起来很麻烦)。

  • 我没有尝试第二种选择,因为我认为这不是解决问题的正确方法。对于第一个选项,我尝试使用Ninject解析订单的类型并将其实例化。我的Ninject模块如下所示:
    private class OrdersService : NinjectModule
    {
        public override void Load()
        {
            Bind<Order>().To<OrderBottling>();
            Bind<Order>().To<OrderFinishing>();
        }
    }
    

    我已经尝试通过Ninject的Get<>()方法获得一种类型,但是它告诉我,这不仅仅是一种解析类型的方法。因此,我了解该模块的实现不正确。我也尝试过对两种类型都实现这种形式:Bind<Order>().To<OrderBottling>().WithPropertyInject("OrderTypeId", 2);,但是它有相同的问题...实现此模块的正确方法是什么?

    我也尝试过使用MvcContrib Model Binder。我已经做到了:
    [DerivedTypeBinderAware(typeof(OrderBottling))]
    [DerivedTypeBinderAware(typeof(OrderFinishing))]
    public abstract partial class Order { }
    

    Global.asax.cs上,我做到了:
    protected void Application_Start()
    {
        AreaRegistration.RegisterAllAreas();
    
        RegisterRoutes(RouteTable.Routes);
    
        ModelBinders.Binders.Add(typeof(Order), new DerivedTypeModelBinder());
    }
    

    但这会引发异常:System.MissingMethodException:无法创建抽象类。因此,我认为活页夹不是或不能解析为正确的类型。

    非常感谢!

    编辑:首先,感谢Martin和Jason的回答,并感谢您的延迟!我尝试了两种方法,并且都有效!我将马丁的回答标记为正确,因为它更灵活并且可以满足我的项目的某些需求。具体来说,每个请求的ID都存储在数据库中,如果我仅在一个地方(数据库或类)更改ID,则将它们放在类上会破坏软件。在这一点上,马丁的方法非常灵活。

    @马丁:在我的代码上,我改变了这一行
    var concreteType = Assembly.GetExecutingAssembly().GetType(concreteTypeValue.AttemptedValue);
    


    var concreteType = Assembly.GetAssembly(typeof(Order)).GetType(concreteTypeValue.AttemptedValue);
    

    因为我的类(class)在另一个项目上(因此在另一个程序集上)。我之所以分享这一点,是因为它似乎比只获取不能解析外部程序集类型的执行程序集更灵活。在我的情况下,所有订单类都在同一程序集上。这既不是更好也不是一个神奇的公式,但是我认为分享这个有趣;)

    最佳答案

    之前我曾尝试做类似的事情,但得出的结论是,没有内置的东西可以处理这个问题。

    我选择的选项是创建自己的模型绑定(bind)程序(尽管它是从默认继承的,所以它没有太多代码)。它寻找名称为xxxConcreteType的类型的回发值,其中xxx是绑定(bind)的另一种类型。这意味着必须使用要绑定(bind)的类型的值将字段回发。在这种情况下,OrderConcreteType的值为OrderBottling或OrderFinishing。

    另一种选择是使用UpdateModel或TryUpdateModel并从方法中省略参数。您需要先确定要更新的模型类型(通过参数或其他方式),然后事先实例化该类,然后可以使用这两种方法来弹出它

    编辑:

    这是代码。

    public class AbstractBindAttribute : CustomModelBinderAttribute
    {
        public string ConcreteTypeParameter { get; set; }
    
        public override IModelBinder GetBinder()
        {
            return new AbstractModelBinder(ConcreteTypeParameter);
        }
    
        private class AbstractModelBinder : DefaultModelBinder
        {
            private readonly string concreteTypeParameterName;
    
            public AbstractModelBinder(string concreteTypeParameterName)
            {
                this.concreteTypeParameterName = concreteTypeParameterName;
            }
    
            protected override object CreateModel(ControllerContext controllerContext, ModelBindingContext bindingContext, Type modelType)
            {
                var concreteTypeValue = bindingContext.ValueProvider.GetValue(concreteTypeParameterName);
    
                if (concreteTypeValue == null)
                    throw new Exception("Concrete type value not specified for abstract class binding");
    
                var concreteType = Assembly.GetExecutingAssembly().GetType(concreteTypeValue.AttemptedValue);
    
                if (concreteType == null)
                    throw new Exception("Cannot create abstract model");
    
                if (!concreteType.IsSubclassOf(modelType))
                    throw new Exception("Incorrect model type specified");
    
                var concreteInstance = Activator.CreateInstance(concreteType);
    
                bindingContext.ModelMetadata = ModelMetadataProviders.Current.GetMetadataForType(() => concreteInstance, concreteType);
    
                return concreteInstance;
            }
        }
    }
    

    将操作方法​​更改为如下所示:
    public ActionResult Create([AbstractBind(ConcreteTypeParameter = "orderType")] Order order) { /* implementation ommited */ }
    

    您需要在您的 View 中添加以下内容:
    @Html.Hidden("orderType, "Namespace.xxx.OrderBottling")
    

    关于c# - ASP.NET MVC 3 : DefaultModelBinder with inheritance/polymorphism,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5460081/

    相关文章:

    c# - Entity Framework Code First POCO 以及为什么/为什么不建立关系的原因?

    c# - 为什么图像在我的按钮内仅部分可见?

    javascript - 如何实现javascript继承

    java - java中如何定义子类(基本继承)

    c# - 具有 IEnumerable<T> 对象的显示模板的文件名

    c# - Membership.IsApproved 值未保存

    Android 用户界面继承

    c# - 从基接口(interface)继承的接口(interface)

    javascript - 使用 jquery 检查和取消选中具有动态 id 和类的复选框

    c# - 获取 viewmodel 属性的 modelstate 键