深入理解java虚拟机·读书笔记(一)

虚拟机类加载机制

时间:20180222


需要弄明白的问题:

类是如何转换为字节码、java虚拟机是如何加载这些字节码文件、加载完字节码文件之后虚拟机内部又是如何进行处理的?

虚拟机 把 Class文件 加载到 内存--->并对数据进行 校验、解析、初始化,最终形成可以被虚拟机直接使用的java类型

类的生命周期:

深入理解java虚拟机·读书笔记(一)_第1张图片
image

1.什么时候开始类的加载过程?java虚拟机规范中并没有给出明确的规定,具体交给虚拟机来自由把握。

2.java虚拟机规范对类的初始化过程进行明确的规定:

深入理解java虚拟机·读书笔记(一)_第2张图片
image

以上称为主动引用,除此之外,所有引用类的方式都不会触发初始化,成为被动引用。

其中: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没有初始化,需要先初始化父类。

以下为被动引用:

深入理解java虚拟机·读书笔记(一)_第3张图片
image

测试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

你可能感兴趣的:(深入理解java虚拟机·读书笔记(一))