画图详解java中类的加载和初始化过程(jdk1.8)
首先上一张类的加载和初始化图
然后对上面的图做详细解释
1.在执行main方法之前会先将PersonTest类和Person类依次加载到方法区,即类中的所有成员变量和方法都会进入方法区。这里加载到方法区的实际上是编译之后的字节码文件,这里为了演示方便直接使用原始代码表示。
2.将类加载到方法区之后,会优先将类的静态成员(包括静态成员变量和静态成员方法)加载到静态区,即将PersonTest类中的main方法和Person类中的静态成员变量classname加载到静态区。这里会对静态成员变量赋默认值,由于classname是String类型,所以默认值就是null。
3.开始执行main方法,这里JVM会将main方法复制一份放入栈区开始执行。
3.1.开始执行main方法的第一行代码,System.out.println(Person.classname)。由于classname是静态成员变量,这个时候JVM就会到静态区取找Person类的静态成员变量classname,找到之后就输出classname的值,由于默认值是null,所以打印的结果就是null。
从这里就可以看出静态是优先于非静态存在的,因为这里还没有创建对象,但是可以直接打印classname的值。现在再来回头理解,说静态方法无法调用非静态成员变量的原因就更加清晰了。
3.2.开始执行main方法的第二行代码,Person p = new Person()。这个时候JVM就会去堆内存开辟一块空间,开辟完成之后这块内存空间就有对应的地址了。然后类的成员变量就会进入该内存空间,即name属性就会被复制一份放入该堆内存中,然后赋默认值,即null。给变量赋默认值之后,就会开始执行构造方法,这个时候构造方法就会被复制一份放入栈区,然后开始执行构造方法体中的内容。构造方法执行完之后,属于构造方法的内存空间就会被销毁。到这里p对象就构建完成了。
3.3.开始执行main方法的第三行代码,p.show()。这个时候show方法会被复制一份放入栈区,然后开始执行show方法中的内容。等show方法执行完成之后,属于show方法的内存空间就会被销毁。到这里main方法里的内容也全部执行完毕了,所以属于main方法的内存空间也会被销毁。随即,Person类型的对象引用p指向堆内存空间的对象也会被断开,到这里整个程序就结束了。