在视图中访问动态匿名类型时的RuntimeBinderException

时间:2013-11-12 17:53:16

标签: c# asp.net-mvc asp.net-mvc-3 dynamic anonymous-types

我在学习/修补asp.net时遇到了一个奇怪的异常现象。

我试图显示这样的部分视图:

@Html.Partial("_PartialView", new { Action = "Foo" })

当我尝试使用

访问操作时
// Throws Microsoft.Csharp.RuntimeBinder.RuntimeBinderException
string throwsException = Model.Action; 

带有消息

的RuntimeBinderException
  

'object'不包含'Action'的定义

被抛出 奇怪的是,这条线很好用:

// This line works fine
string works = ((Type)Model.GetType()).GetProperty("Action").GetValue(Model);

这种行为让我很困惑,我宁愿避免使用这种解决方法。此外,我不认为问题是anonymous types being internal,因为VS2013中ASP.NET项目的MVC模板成功实现了这一点:

enter image description here

那么这里发生了什么?

3 个答案:

答案 0 :(得分:5)

这个问题的答案可以在这里找到:http://www.heartysoft.com/ashic/blog/2010/5/anonymous-types-c-sharp-4-dynamic

来自优秀的博客文章:

Anonymous Types are Internal

调用Model.Action失败的原因是Model的类型信息在运行时不可用。它不可用的原因是因为匿名类型不公开。当该方法返回该匿名类型的实例时,它返回一个System.Object,该System引用一个匿名类型的实例 - 一个类型的信息不可用于主程序。动态运行时尝试在对象上查找名为Action的属性,但无法从其拥有的类型信息中解析它。因此,它会引发异常。

答案 1 :(得分:3)

  

那么这里发生了什么?

您的部分视图输入较弱。您没有@model定义。所以默认情况下它是object,显然没有Action属性。

解决此问题的正确方法是定义视图模型:

public class MyViewModel
{
    public string Action { get; set; }
}

您的部分视图将被强类型化为:

@model MyViewModel
@{
    string throwsException = Model.Action; 
}

将由主视图传递:

@Html.Partial("_PartialView", new MyViewModel { Action = "Foo" })

另一种可能性(我个人不喜欢,因为它依赖于运行时绑定)是在局部视图中使用动态模型:

@model dynamic
@{
    string throwsException = Model.Action; 
}

然后你可以在调用它时传递一个匿名对象:

@Html.Partial("_PartialView", new { Action = "Foo" })

答案 2 :(得分:1)

以下是使用反射的一些选项。在大多数情况下,性能应该可以忽略不计。

公用事业类

public static class ModelHelper
{
    public static T Data<T>( String key)  
    {
        var html = ((System.Web.Mvc.WebViewPage)WebPageContext.Current.Page).Html;
        var model = html.ViewData.Model;
        return (T)model.GetType().GetProperty(key).GetValue(model) ;
    }
}

<强> view.cshtml

@(Html.Partial("partial",new { Id="InstructorId"}))

paritial.cshtml文件

@model dynamic 
@{ 
    var id = ModelHelper.Data<String>("Id");
}

在view.cshtml中使用更紧凑的parital调用

@{
  var instructorId = "InstructorId";
  var windowTitle = "Window Title";
  var editorPageUrl = "~/View/Editors/Instructors.chstml";
}

@(Html.Partial("partial",new { instructorId, windowTitle, editorPageUrl }))

调整变量以在paritial.cshtml文件中获取推断名称

@model dynamic 
@{ 
    var id = ModelHelper.Data<String>("instructorId");
    var id = ModelHelper.Data<String>("windowTitle");
    var id = ModelHelper.Data<String>("editorPageUrl");
}

创建一个简单的视图模型类并不难,但我不喜欢一次只能用于传递数据的类,所以这对我有用。

您还可以扩展默认视图库

namespace SafetyPlus.Shell.Code
{
    public abstract class ExtendedPageBaseClass<TModel> : WebViewPage<TModel> where TModel : class
    {
        public T Data<T>(String key) 
        {
            return (T)Model.GetType().GetProperty(key).GetValue(Model);
        }

        public Object Data(String key) 
        {
            return Data<Object>(key);
        }
    }
}

在/Views/web.config中注册基类

<pages pageBaseType="SafetyPlus.Shell.Code.ExtendedPageBaseClass">
  ...
</pages>

在您的部分视图中获取数据

@{ 
    var id = Data("Id");
    var idTyped = Data<String>("Id");
}

或使用我建议的扩展方法。

namespace NotYourDefaultNamespace
{
    public static class ModelExtensions
    {
        public static T Data<T>( this Object model, String key)  
        {
            return (T)model.GetType().GetProperty(key).GetValue(model) ;
        }
    }
} 

由于我们从之前的实用方法中找到了模型,因此这个选项确实不会给你带来任何好处。电话会变成。

@{
    var id = Model.Data("Id");
}