So this is the meat of the question: Can Foo.Bar ever return null? To clarify, can ‘_bar’ be set to null after it’s evaluated as non-null and before it’s value is returned?
public class Foo
{
Object _bar;
public Object Bar
{
get { return _bar ?? new Object(); }
set { _bar = value; }
}
}
I know using the following get method is not safe, and can return a null value:
get { return _bar != null ? _bar : new Object(); }
UPDATE:
Another way to look at the same problem, this example might be more clear:
public static T GetValue<T>(ref T value) where T : class, new()
{
return value ?? new T();
}
And again asking can GetValue(…) ever return null? Depending on your definition this may or may not be thread-safe… I guess the right problem statement is asking if it is an atomic operation on value… David Yaw has defined the question best by saying is the above function the equivalent to the following:
public static T GetValue<T>(ref T value) where T : class, new()
{
T result = value;
if (result != null)
return result;
else
return new T();
}
No, this is not thread safe.
The IL for the above compiles to:
This effectively does a load of the
_barfield, then checks its existence, and jumps ot the end. There is no synchronization in place, and since this is multiple IL instructions, it’s possible for a secondary thread to cause a race condition – causing the returned object to differ from the one set.It’s much better to handle lazy instantiation via
Lazy<T>. That provides a thread-safe, lazy instantiation pattern. Granted, the above code is not doing lazy instantiation (rather returning a new object every time until some time when_baris set), but I suspect that’s a bug, and not the intended behavior.In addition,
Lazy<T>makes setting difficult.To duplicate the above behavior in a thread-safe manner would require explicit synchronization.
As to your update:
The getter for the Bar property could never return null.
Looking at the IL above, it
_bar(via ldfld), then does a check to see if that object is not null using brtrue.s. If the object is not null, it jumps, copies the value of_barfrom the execution stack to a local via stloc.0, and returns – returning_barwith a real value.If
_barwas unset, then it will pop it off the execution stack, and create a new object, which then gets stored and returned.Either case prevents a
nullvalue from being returned. However, again, I wouldn’t consider this thread-safe in general, since it’s possible that a call to set happening at the same time as a call to get can cause different objects to be returned, and it’s a race condition as which object instance gets returned (the set value, or a new object).