问题
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?
回答1:
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 classTestLab
. From here,Test
class is referenced. So, control moves toTest
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 ofTest
, so control moves to instance block (In instance block of Test) ofTest
and then to constructor (Default Constructor of Test).Now
static Test ref=new Test();
is complete, so, the class initialization ofTest
continues and reaches the static block (In Static Block of Test). This completes initialization ofTest
.Control reaches back to
TestLab
, nownew Test()
is called. So again In instance block of Test and Default Constructor of Test are printed (class is already initialized, sostatic
fields are not initialized again and static blocks are not executed).Control reaches static block of
TestLab
(In static block of TestLab).
回答2:
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.
回答3:
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.
来源:https://stackoverflow.com/questions/32754136/when-objects-created-with-static-reference-why-do-instance-block-default-cons