虚拟机类加载机制
时间:20180222
需要弄明白的问题:
类是如何转换为字节码、java虚拟机是如何加载这些字节码文件、加载完字节码文件之后虚拟机内部又是如何进行处理的?
虚拟机 把 Class文件 加载到 内存--->并对数据进行 校验、解析、初始化,最终形成可以被虚拟机直接使用的java类型
的
类的生命周期:
1.什么时候开始类的加载过程?java虚拟机规范中并没有给出明确的规定,具体交给虚拟机来自由把握。
2.java虚拟机规范对类的初始化过程进行明确的规定:
以上称为主动引用,除此之外,所有引用类的方式都不会触发初始化,成为被动引用。
其中:1.当初始化一个类的时候,如果其父类还没有进行过初始化,则需要一层层往上找到父类,从最上层的父类开始初始化。2.使用new关键字实例化对象时候、读取或设置一个类的静态字段(被final修饰、已在编译期把结果放入常量池的静态字段除外),以及调用一个类的静态方法的时候。
测试code:
package com.testf;
public class Parent {
static {
System.out.println("Parent 初始化");
}
}
package com.testf;
public class Child extends Parent{
static {
System.out.println("child 初始化....");
}
public static void main(String[] args) {
System.out.println("main 初始化....");
}
}
上述代码执行结果为:
Parent 初始化
child 初始化....
main 初始化....
以上结果:表明当初始化子类Child的时候发现父类Parent没有初始化,需要先初始化父类。
以下为被动引用:
测试code:
package com.testf;
public class Parent {
public static int num = 10;
}
package com.testf;
public class Child extends Parent{
static {
System.out.println("child 初始化....");
}
public static final int a = 10;
}
情况1
package com.testf;
//测试子类调用父类的静态字段(调用父类Parent中静态字段num)
//Child类并不会被加载,即子类不会被加载
public class Main {
static {
System.out.println("Main 类初始化");
}
public static void main(String[] args) {
System.out.println(Child.num);
}
}
结果
Main 类初始化
Parent 初始化
10
情况2
package com.testf;
//测试通过数组定义的引用类
public class Main {
static {
System.out.println("Main 类初始化");
}
public static void main(String[] args) {
Child[] c = new Child[20];
//System.out.println(Child.a);
}
}
结果
Main 类初始化
情况3
package com.testf;
//测试通过调用类常量
public class Main {
static {
System.out.println("Main 类初始化");
}
public static void main(String[] args) {
System.out.println(Child.a);
}
}
结果
Main 类初始化
10