Consider the following code:
using System;
namespace ConsoleApplication2
{
class Program
{
static void Main(string[] args)
{
var square = new Square(4);
Console.WriteLine(square.Calculate());
}
}
class MathOp
{
protected MathOp(Func<int> calc) { _calc = calc; }
public int Calculate() { return _calc(); }
private Func<int> _calc;
}
class Square : MathOp
{
public Square(int operand)
: base(() => _operand * _operand) // runtime exception
{
_operand = operand;
}
private int _operand;
}
}
(ignore the class design; I’m not actually writing a calculator! this code merely represents a minimal repro for a much bigger problem that took awhile to narrow down)
I would expect it to either:
- print “16”, OR
- throw a compile time error if closing over a member field is not allowed in this scenario
Instead I get a nonsensical exception thrown at the indicated line. On the 3.0 CLR it’s a NullReferenceException; on the Silverlight CLR it’s the infamous Operation could destabilize the runtime.
It’s not going to result in a compile-time error because it is a valid closure.
The problem is that
thisis not initialized yet at the time the closure is created. Your constructor hasn’t actually run yet when that argument is supplied. So the resultingNullReferenceExceptionis actually quite logical. It’sthisthat’snull!I’ll prove it to you. Let’s rewrite the code this way:
Guess what this prints? Yep, it’s
true, the closure returnsnullbecausethisis not initialized when it executes.Edit
I was curious about Thomas’s statement, thinking that maybe they’d changed the behaviour in a subsequent VS release. I actually found a Microsoft Connect issue about this very thing. It was closed as “won’t fix.” Odd.
As Microsoft says in their response, it is normally invalid to use the
thisreference from within the argument list of a base constructor call; the reference simply does not exist at that point in time and you will actually get a compile-time error if you try to use it “naked.” So, arguably it should produce a compile error for the closure case, but thethisreference is hidden from the compiler, which (at least in VS 2008) would have to know to look for it inside the closure in order to prevent people from doing this. It doesn’t, which is why you end up with this behaviour.