public class TestLab {
static Test aStatic=new Test();
public static void main(String[] args) {
TestLab obj=new TestLab();
}
static{
System.out.println("In static block of TestLab");
}
}
public class Test {
static Test ref=new Test();
Test()
{
System.out.println("Default Constructor of Test");
}
static
{
System.out.println("In Static Block of Test");
}
{
System.out.println("In instance block of Test");
}
}
Normally the static blocks are executed first during class loading. When the above example is executed, the following output is received:
In instance block of Test
Default Constructor of Test
In Static Block of Test
In instance block of Test
Default Constructor of Test
In static block of TestLab
Why does the instance block and Default Constructor of Test class gets executed before the static block of Test Class?
Ok. static
fields / blocks are set / executed during class initialization. They are executed in the order that they appear in the code.
So, after class TestLab
is loaded, when it is getting initialized, the following things happen :
static Test aStatic=new Test();
==>
Called as part of initialization of class TestLab
. From here, Test
class is referenced. So, control moves to Test
class.
static Test ref=new Test();
==> i.e, first line of Test class (during its initialization phase) is executed. This line involves creating a new instance of Test
, so control moves to instance block (In instance block of Test) of Test
and then to constructor (Default Constructor of Test).
Now static Test ref=new Test();
is complete, so, the class initialization of Test
continues and reaches the static block (In Static Block of Test). This completes initialization of Test
.
Control reaches back to TestLab
, now new Test()
is called. So again In instance block of Test and Default Constructor of Test are printed (class is already initialized, so static
fields are not initialized again and static blocks are not executed).
Control reaches static block of TestLab
(In static block of TestLab).
When the type is initialized, all the static initializers and all the static field initializers are executed, in textual order. From JLS 12.4.2:
Next, execute either the class variable initializers and static initializers of the class, or the field initializers of the interface, in textual order, as though they were a single block.
In other words, this code gets executed:
ref = new Test();
System.out.println("In Static Block of Test");
That first line creates an instance... which requires the instance initializer to be run. All of that instance initialization happens before control returns to the type initialization part - i.e. before the line from the static initializer runs.
If you move the field declaration to after the static initializer, you'll see the opposite results.
Usually static variables/blocks
will be initialized in the order they are defined, here you have marked aStatic
as static
. It will try to create the instance of Test by invoking constructor, but as instance block is provided it will get executed and then the constructor and eventually the static block.