为什么以下C#代码在可空的DateTime初始化时抛出NullReferenceException?

时间:2013-09-13 07:29:04

标签: c# nullreferenceexception nullable

这是一些简单的代码:

    static void Main(string[] args)
    {
        var coll = new List<string> {"test", "test2", "test3"};

        var filteredColl = coll.Select(x => x).ToList();

        if (!filteredColl.Any())
        {
            DateTime? date = new DateTime();

            filteredColl = coll.Where(x => date.GetValueOrDefault().Date.ToString(CultureInfo.InvariantCulture) == x).Select(x => x).ToList();
        }
    }

问题是,以下步骤使NullReferenceException崩溃的原因

1)断点到if

Breakpoint

2)设置下一个执行点:

Execution point

3)尝试继续使用F10:

Exception

如果我注释掉代码的最后一行,它就不会崩溃。

更新:这是堆栈跟踪:

System.NullReferenceException was unhandled   HResult=-2147467261  
Message=Object reference not set to an instance of an object.  
Source=ConsoleApplication28   StackTrace:
       at ConsoleApplication28.Program.Main(String[] args) in Program.cs: line 21
       at System.AppDomain._nExecuteAssembly(RuntimeAssembly assembly, String[] args)
       at System.AppDomain.ExecuteAssembly(String assemblyFile, Evidence assemblySecurity, String[] args)
       at Microsoft.VisualStudio.HostingProcess.HostProc.RunUsersAssembly()
       at System.Threading.ThreadHelper.ThreadStart_Context(Object state)
       at System.Threading.ExecutionContext.RunInternal(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
       at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
       at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state)
       at System.Threading.ThreadHelper.ThreadStart()   InnerException:

2 个答案:

答案 0 :(得分:9)

这是在声明捕获的变量范围的上下文中移动执行点的副作用。将其报告为IDE错误是合理的,但修复并不是一件容易的事。基本上,date不是变量 - 它是捕获上下文中的字段,因为lambda。编译器本质上是这样做的:

if (!filteredColl.Any())
{
    var ctx = new SomeCaptureContext(); // <== invented by the compiler
    ctx.date = new DateTime();

    filteredColl = coll.Where(ctx.SomePredicate).Select(x => x).ToList();
}

其中SomePredicate是:

class SomeCaptureContext {
    public DateTime? date; // yes, a public field - needs to support 'ref' etc
    public bool SomePredicate(string x) // the actual name is horrible
    {
        return this.date.GetValueOrDefault()
              .Date.ToString(CultureInfo.InvariantCulture) == x;
    }
}

这里的问题是当你将执行位置拖到:

DateTime? date = new DateTime();

实际(用IL术语)将它拖到以下行:

ctx.date = new DateTime();

紧接在此之前的捕获上下文行,即

var ctx = new SomeCaptureContext();

从未执行过,因此ctxnull。因此NullReferenceException

将此作为一个错误记录是合理的,但它是一个微妙的 - 你不一定总是想要拖动执行上下文来初始化捕获上下文 - 它会有是“如果他们是null”。

答案 1 :(得分:0)

嗯......真的很奇怪。 你有引用中的mscorlib和使用中的系统吗? 也许你有一个名为DateTime的另一个类,它会覆盖System.DateTime。