If a type has no static constructor, field initializers will execute just prior to the type being used— or anytime earlier at the whim of the runtime
Why this code :
void Main()
{
"-------start-------".Dump();
Test.EchoAndReturn("Hello");
"-------end-------".Dump();
}
class Test
{
public static string x = EchoAndReturn ("a");
public static string y = EchoAndReturn ("b");
public static string EchoAndReturn (string s)
{
Console.WriteLine (s);
return s;
}
}
yields :
-------start-------
a
b
Hello
-------end-------
while this code :
void Main()
{
"-------start-------".Dump();
var test=Test.x;
"-------end-------".Dump();
}
yields
a
b
-------start-------
-------end-------
The order of a
and b
is understood. but why dealing with static method
is different than static field
.
I mean why the start and end lines are in different locations with static methods vs static fields ? I mean - in both situation he's got to initialize those fields...so why ?
( I know I can add static ctor which make it to be the same - but Im asking about this particular situation. )
(p.s. Dump() is just like console.write)
The behavior of the release JIT is (from 4.0 IIRC) not to run the static initializer unless the method you are calling touches static fields. This can mean the static fields are not initialized. If I run your first code in release outside of the debugger, I get:
If I run it with the debugger attached (release), or in a debug build (with or without debugger attached), I get:
So far so interesting. For why you get:
it looks like the per-method JIT is essentially taking responsibility for running the static constructor in this scenario. You can see this by adding:
which will print (even in release without the debugger)
so the possibility of accessing fields is key. If we change
Test.x
to be a call to a method that doesn't access fields (and remove theNeverTrue()
thing), then we get no output whatsoever.So: in some versions of the CLI, the execution of static initializers may be deferred to the JIT-step of methods that contain mentions to any field (it doesn't check whether that field has an initializer).
We can even create instances of objects without running the static initializer, as long as we don't touch static fields:
with:
prints just (release, no debugger):
HOWEVER! We should not build anything that depends on this timing:
This is with
.NET 4.0
If a type has no static constructor but static fields with initialization, the compiler creates a type constructor and puts the initialization inside it.
Results in the following IL (Just the cctor part)
Also, according to
CLR via C#
, the JIT compiler checks each method beforehand, what types have a static constructor. If the static constructor has not been called, the JIT compiler calls it.That would explain the difference between the two code fragments.
@Comment
If you move one field initializer into a user-defined type constructor, the compiler will move the other field you initialize on the class level into the type constructor as well.
Results in the same IL as above. So there basically is no difference between your own type constructor and the one generated by the compiler (There can only be one anyway).
The time when static constuctor will be called is not guaranted, so for the programm it is like Undefined Behavior in C++. Nobody should rely on the sequence of static constructor calls. For example if you compile the programm under release you will see that static counstructor called in the same time in both cases.