关于Java里static的一些思考
Java Static
最近帮朋友做一个项目的时候想到了一个问题,那就是static到底是个什么东西?结果在参考了网上的一些内容之后,发现了一些更有趣的东西。那就是java类被虚拟机加载时的加载顺序。话不多说,先看一下下面这个经典的面试题例子。
public class Test {
Person person = new Person("Test");
static{
System.out.println("test static");
}
public Test() {
System.out.println("test constructor");
}
public static void main(String[] args) {
new MyClass();
}
}
class Person{
static{
System.out.println("person static");
}
public Person(String str) {
System.out.println("person "+str);
}
}
class MyClass extends Test {
Person person = new Person("MyClass");
static{
System.out.println("myclass static");
}
public MyClass() {
System.out.println("myclass constructor");
}
}
我们可以看到,代码并不算麻烦,那么他到底哪里有趣吸引了我呢?就是运行时的输出结果:
test static
myclass static
person static
person Test
test constructor
person MyClass
myclass constructor
Process finished with exit code 0
也许有很多人会对这个输出感到疑问。各位看官勿急,先看一下下面我总结的这两张流程图。这两张图基本能展示为什么会按照上面的输出结果输出。
规则流程图1:含有public static void main方法的public类
规则流程图2:不含有public static void main方法的类
下面附上另一位大神对这道题的解释:
类似地,我们还是来想一下这段代码的具体执行过程。首先加载Test类,因此会执行Test类中的static块。接着执行new MyClass(),而MyClass类还没有被加载,因此需要加载MyClass类。在加载MyClass类的时候,发现MyClass类继承自Test类,但是由于Test类已经被加载了,所以只需要加载MyClass类,那么就会执行MyClass类的中的static块。在加载完之后,就通过构造器来生成对象。而在生成对象的时候,必须先初始化父类的成员变量,因此会执行Test中的Person person = new Person(),而Person类还没有被加载过,因此会先加载Person类并执行Person类中的static块,接着执行父类的构造器,完成了父类的初始化,然后就来初始化自身了,因此会接着执行MyClass中的Person person = new Person(),最后执行MyClass的构造器。