5

我注意到 RazorEngine.Compile() 似乎将匿名类型与其他类型不同。例如,考虑以下代码:

public static void Main()
{
    try {
    var model = new { s = default(string) };
    RazorEngine.Razor.Compile("@Model.s.Length", model.GetType(), "a");
    RazorEngine.Razor.Run(model, "a");
    } catch (Exception ex) {
        Console.WriteLine(ex); // RuntimeBinderException (Cannot perform runtime binding on a null reference)
    }

    try 
    {
    var model = "";
    RazorEngine.Razor.Compile(@"@Model.Length", model.GetType(), "b");
    RazorEngine.Razor.Run(default(string), "b");
    } catch (Exception ex) {
        Console.WriteLine(ex); // NullReferenceException
    }

    try 
    {
    var model = Tuple.Create(default(string));
    RazorEngine.Razor.Compile(@"@Model.Item1.Length", model.GetType(), "c");
    RazorEngine.Razor.Run(model, "c");
    } catch (Exception ex) {
        Console.WriteLine(ex); // NullReferenceException
    }

        try 
    {
    var model = new Internal();
    RazorEngine.Razor.Compile(@"@Model.S.Length", model.GetType(), "d");
    RazorEngine.Razor.Run(model, "d");
    } catch (Exception ex) {
        Console.WriteLine(ex); // TemplateCompilationException (type Internal is not visible)
    }
}

internal class Internal {
    public string S { get; set; }
}

我的理解是:匿名类型是内部的,所以通常 Razor 不会处理它们。但是,Razor 通过生成动态模板来为匿名类型提供特殊支持。

因此,我有两个问题:(1)我对这种行为的理解是否正确?(2) 有没有办法让剃刀为匿名模型输出强类型模板?

4

2 回答 2

3

我的想法是让 Razor 程序集成为您程序集的朋友程序集(从而使内部部件可见),但这没有奏效。在 Razor 源代码中,我们可以看到以下代码(in CompilerServiceBase.cs):

 if (modelType != null)
 {
     if (CompilerServices.IsAnonymousType(modelType))
     {
         type.CustomAttributes.Add(new CodeAttributeDeclaration(new CodeTypeReference(typeof(HasDynamicModelAttribute))));
     }
 }

然后稍后(在TemplateBaseOfT.cs):

HasDynamicModel = GetType().IsDefined(typeof(HasDynamicModelAttribute), true);

稍后在同一文件中使用:

if (HasDynamicModel && !(value is DynamicObject) && !(value is ExpandoObject))
    model = new RazorDynamicObject { Model = value };
else
    model = value;

所以是的 - 你的理解是正确的。此外,我们可以看到使内部结构可见是不够的,因为 Razor 将任何匿名类型视为动态 ( RazorDynamicObjectextends DynamicObject)。如果包含程序集的内部可见,您可以尝试修补 Razor 代码以不将匿名类型视为动态类型。

不过,在这种情况下,我认为代码必须发出一个新的公共类型,该类型包含与模型的匿名类型相同的属性,以便 Razor 生成的代码可以实例化该类型的实例。或者,您可以使用此处描述的 hack来允许方法返回匿名类型的实例。

于 2013-02-14T17:04:34.840 回答
1

动态用于允许匿名类型。当它编译模型时,它不能引用匿名类型,因为它们在您的程序集中。考虑到这一点,Razor 解析器正在尝试生成一个继承自的模板TemplateBase<T>以提供强类型支持。这就是我们使用的TemplateBase<dynamic>原因,因为这意味着我们将对这些属性的访问权委托给 DLR,但是您会失去静态类型类的好处。

于 2013-06-26T15:23:39.397 回答