22

我一直在查看与 Razor View 引擎关联的调试器中的一些代码,我注意到某些类型出现在 Debugger 中,在类型名称的末尾带有一个尾随点字符,例如:

{南希.ViewEngines.Razor.RazorViewEngine.}

有谁知道这表明什么?在对象上指定强制转换时使用它不是有效的语法,所以我对它在调试器中指示的内容很感兴趣。

编辑:根据@Damien_The_Unbeliever 的要求,调试器中变量的屏幕截图:

调试图像

我正在查看的代码:

public TCompiledView GetOrAdd<TCompiledView>(
            ViewLocationResult viewLocationResult, Func<ViewLocationResult, TCompiledView> valueFactory)
        {
            TCompiledView compiledView = default(TCompiledView);
            compiledView = (TCompiledView)this.cache.GetOrAdd(viewLocationResult, x => valueFactory(x));

为了提供更多背景信息,我们正在尝试将日志记录添加到我们的 Nancy View Cache 以调查 Razor Views 引发编译错误的间歇性问题,但这与问题并不真正相关。

4

1 回答 1

12

我已经看到当变量/值实际上是编译器生成的类型时会发生这种情况(例如,用于保存由 lambda、异步、迭代器等捕获的“局部变量”)。调试器(在不同的地方)似乎无法显示实际的类名。


例如这个示例程序:

class Program
{
    static void Main(string[] args)
    {
        var p = new Program();
        p.DoStuff();
    }

    void DoStuff()
    {
        int i = 19;
        Expression<Func<int>> j = () => i + 10;
        var k = (((j.Body as BinaryExpression).Left as MemberExpression).Expression as ConstantExpression).Value;
        Console.ReadLine();
    }
}

使用断点 on Console.ReadLine(),您会发现k类的类型看起来Program.Program+<>_DisplayClass0


Jeppe 的加法:这个例子是上面的一个轻微的简化,避免了表达式树。Target查看将成为生成类的实例的委托实例。为了比较,还查看了一个迭代器块类型:

using System;
using System.Collections.Generic;

static class Program
{
  static void Main()
  {
    int i = 19; // to be captured by lambda, will become field on a generated class
    Func<int> f = () => i;
    var target = f.Target;  // when debugging type looks like "Program."
    Console.WriteLine(target.GetType().ToString()); // writes "Program+<>c__DisplayClass1"

    var seq = GetSeq();  // when debugging type looks like "Program.GetSeq"
    Console.WriteLine(seq.GetType().ToString()); // writes "Program+<GetSeq>d__3"
  }

  static IEnumerable<int> GetSeq() // returns "state machine" (iterator block)
  {
    yield return 42;
  }
}
于 2013-08-05T14:07:27.387 回答