jjzjj

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

coder 2024-05-18 原文

首先,对于大型文章(我首先尝试做一些研究)以及针对同一问题的技术混合(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# - ASP.NET MVC 3 : DefaultModelBinder with inheritance/polymorphism的更多相关文章

    1. c# - 如何在 ruby​​ 中调用 C# dll? - 2

      如何在ruby​​中调用C#dll? 最佳答案 我能想到几种可能性:为您的DLL编写(或找人编写)一个COM包装器,如果它还没有,则使用Ruby的WIN32OLE库来调用它;看看RubyCLR,其中一位作者是JohnLam,他继续在Microsoft从事IronRuby方面的工作。(估计不会再维护了,可能不支持.Net2.0以上的版本);正如其他地方已经提到的,看看使用IronRuby,如果这是您的技术选择。有一个主题是here.请注意,最后一篇文章实际上来自JohnLam(看起来像是2009年3月),他似乎很自在地断言RubyCL

    2. C# 到 Ruby sha1 base64 编码 - 2

      我正在尝试在Ruby中复制Convert.ToBase64String()行为。这是我的C#代码:varsha1=newSHA1CryptoServiceProvider();varpasswordBytes=Encoding.UTF8.GetBytes("password");varpasswordHash=sha1.ComputeHash(passwordBytes);returnConvert.ToBase64String(passwordHash);//returns"W6ph5Mm5Pz8GgiULbPgzG37mj9g="当我在Ruby中尝试同样的事情时,我得到了相同sha

    3. 基于C#实现简易绘图工具【100010177】 - 2

      C#实现简易绘图工具一.引言实验目的:通过制作窗体应用程序(C#画图软件),熟悉基本的窗体设计过程以及控件设计,事件处理等,熟悉使用C#的winform窗体进行绘图的基本步骤,对于面向对象编程有更加深刻的体会.Tutorial任务设计一个具有基本功能的画图软件**·包括简单的新建文件,保存,重新绘图等功能**·实现一些基本图形的绘制,包括铅笔和基本形状等,学习橡皮工具的创建**·设计一个合理舒适的UI界面**注明:你可能需要先了解一些关于winform窗体应用程序绘图的基本知识,以及关于GDI+类和结构的知识二.实验环境Windows系统下的visualstudio2017C#窗体应用程序三.

    4. ruby - 在 ASP 页面上 Mechanize 中断 - 2

      require'mechanize'agent=Mechanize.newlogin=agent.get('http://www.schoolnet.ch/DE/HomeDE.htm')agent.clicklogin.link_withtext:/Login/然后我得到Mechanize::UnsupportedSchemeError。 最佳答案 Mechanize不支持javascript但您可以将搜索字段添加到表单并为其分配搜索词并使用mechanize提交表单form=page.forms.firstform.add_fie

    5. ruby - "messy-polymorphism"反模式 - 2

      在相当大的Ruby应用程序中,我们会遇到这样一种情况,即给定的对象由一些东西来标识:例如名称和ID。这些值类型中的每一个都有不同的用途,因此并不完全等同(id和name存在于不同的地方)。因此,我们最终在应用程序周围传递了各种值(ID、名称和对象)。这种情况至少在某种程度上似乎是个问题,因为我们已经被错误所困扰,这些错误涉及不清楚应该将什么类型传递给给定函数。事实上,我记得多年来在许多应用程序中看到过类似的问题代码,尽管我再也没有给它一个具体的名称。作为一种无类型语言,Ruby不允许像C++那样使用经典的基于类型的多态函数。作为解决方法,一位同事经常使用这种代码:definitiali

    6. c# - C# 中的 Flatten Ruby 方法 - 2

      我如何做Ruby方法"Flatten"RubyMethod在C#中。此方法将锯齿状数组展平为一维数组。例如:s=[1,2,3]#=>[1,2,3]t=[4,5,6,[7,8]]#=>[4,5,6,[7,8]]a=[s,t,9,10]#=>[[1,2,3],[4,5,6,[7,8]],9,10]a.flatten#=>[1,2,3,4,5,6,7,8,9,10 最佳答案 递归解决方案:IEnumerableFlatten(IEnumerablearray){foreach(variteminarray){if(itemisIEnume

    7. ruby - 可以像在 C# 中使用#region 一样在 Ruby 中使用 begin/end 吗? - 2

      我最近从C#转向了Ruby,我发现自己无法制作可折叠的标记代码区域。我只是想到做这种事情应该没问题:classExamplebegin#agroupofmethodsdefmethod1..enddefmethod2..endenddefmethod3..endend...但是这样做真的可以吗?method1和method2最终与method3是同一种东西吗?还是有一些我还没有见过的用于执行此操作的Ruby惯用语? 最佳答案 正如其他人所说,这不会改变方法定义。但是,如果要标记方法组,为什么不使用Ruby语义来标记它们呢?您可以使用

    8. c# - Ruby 等效于 C# Linq 聚合方法 - 2

      什么是Linq聚合方法的ruby​​等价物。它的工作原理是这样的varfactorial=new[]{1,2,3,4,5}.Aggregate((acc,i)=>acc*i);每次将数组序列中的值传递给lambda时,变量acc都会累积。 最佳答案 这在数学以及几乎所有编程语言中通常称为折叠。它是更普遍的变形概念的一个实例。Ruby从Smalltalk中继承了这个特性的名称,它被称为inject:into:(像aCollectioninject:aStartValueinto:aBlock一样使用。)所以,在Ruby中,它称为inj

    9. c# - 先学什么? - 2

      关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭8年前。Improvethisquestion几年前我去学校学习编程,毕业后我找到了一份系统管理方面的工作,这就是我职业生涯的方向。我想重新开始某种开发,并且一直在“玩”C#和ASP.NET,但我已经听到很多关于其他"new"语言的讨论(新的意思是它们是新的)我)喜欢Ruby和F#。我想我想知道我是否在浪费时间学习主要的MS语言,而不是成为一名通才。很长一段时间没有离开开发社区(如果我曾经离开过的话)让我在潮流中挣扎,我不想落在时代的

    10. c# - 在 C# 中重现 Ruby OpenSSL private_encrypt 输出 - 2

      我有一个简单的Ruby脚本,我用它在某些HTTPheader上执行private_encrypt以签署要发送到ruby​​RESTAPI的Web请求,该API会根据Base64编码字符串测试Base64编码字符串生成而不是解码Base64和解密数据然后测试原始字符串。我使用的脚本是require"openssl"require"base64"path_to_cert=ARGV[0].dupplain_text=Base64.decode64(ARGV[1].dup)private_key=OpenSSL::PKey::RSA.new(File.read(path_to_cert))pu

    随机推荐