类加载、连接、初始化

1.加载

类的加载指的是将类的.class文件中二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构。在这个阶段,会执行类中声明的静态代码块。也就是类中的静态块执行时不需要等到类的初始化。

1.1 加载class文件的方式

             类加载、连接、初始化_第1张图片

类加载的最终产品是位于堆区中的class对象,Class对象封装了类在方法区内的数据结构,并向Java程序员提供了访问方法区内的数据机构的接口.

我们可以通过类名.class来获取一个类的类型的引用,通过new 类名().getClass()来获取一个实例变量的类的引用

1.2类的加载机制

JDK1.2开始类加载采用父亲委托机制。除了Java虚拟机自带的根类加载器以外,其余的类加载器都有且只有一个父加载器。当Java程序使用加载器加载某个类时,加载器会首先委托自己的父加载器去加载该类,若父加载器能加载,则由父加载器完成加载任务,否则才由自定义加载器去加载

类加载、连接、初始化_第2张图片

同时,所有能成功返回Class对象的引用的类加载器(包括定义类加载器,即包括定义类加载器和它下面的所有子加载器)都被称为初始类加载器。 假设loader1实际加载了Sample类,则loader1Sample类的定义类加载器,loader1包括其子类加载器都是初始化加载器。这里并不是继承关系,其实是一种包装或者组合关系。

2.连接

类加载完成后就进入了类的连接阶段,连接阶段主要分为三个过程分别是:验证,准备和解析。在连接阶段,主要是将已经读到内存的类的二进制数据合并到虚拟机的运行时环境中去

2.1验证

类加载、连接、初始化_第3张图片

2.2准备

这个阶段主要是为对象和变量分配内存,并为类设置初始值(方法区中)

对于static类型变量在这个阶段会为其赋值为默认值,比如public static int v=5,在这个阶段会为其赋值为v=0,而对于static final类型的变量,在准备阶段就会被赋值为正确的值

2.3解析

在这个阶段会将符号引用转换成直接引用 
原来的符号引用仅仅是一个字符串,而引用的对象不一定被加载,直接引用只的是将引用对象的指针或者地址偏移量指向真正的对象,将字符串所指向的对象加载到内存中。

3.初始化

在这个阶段主要执行类的构造方法。并且为静态变量赋值为初始值,执行静态块代码。

类的初始化步骤 
1、假如这个类还没有被加载和连接,那就先进行加载和连接 
2、假如类存在直接的父类,并且这个父类还没有被初始化,那就先初始化它的父类 
3、假如类中存在初始化语句时,那就依次执行这些初始化语句。 
类的初始化时机 
所有的Java类只有在对类的首次主动使用时才会被初始化。主动使用的情况有六中,其他情况都属于被动使用: 
1 创建类的实例 
2、访问某个类或接口的静态变量,或者对该静态变量赋值 
3、调用类的静态方法 
4、反射(Class.fotName 
5、初始化一个类的子类 
6Java虚拟机启动时被标明为启动类的类(main方法所在的类) 
注意:1、当Java虚拟机初始化一个类时,要求他的所有父类都已经被初始化,但是这条规则并不适合接口。在初始化一个类或接口时,并不会先初始化它所实现的接口。 
2
、只有当程序访问的静态变量或静态方法确实在当前类或当前接口中定义时,才可以认为是对类或接口的主动使用。如果静态方法或变量在parent中定义,从子类进行调用,则不会初始化子类。 

类加载、连接、初始化_第4张图片

程序中对子类的主动使用会导致父类被初始化,但是对父类的主动使用不会导致子类的初始化(不能说生成一个Object类的对象就导致系统中所有的子类都会被初始化)

只有当程序访问的静态变量或静态方法确实在当前类或当前接口中定义时,才可以认为是类或接口的主动使用,下面举例说明

class Parent3{

    static int a = 3;

    static{

       System.out.println("Parent3 static block");

    }

    static void doSomething(){

       System.out.println("do something");

    }

}

 

class Child3 extends Parent3{

    static{

       System.out.println("child3 static block");

    }

}

public class Test6 {

    public static void main(String[] args) {

       System.out.println(Child3.a);

       Child3.doSomething();

    }

}

执行结果

/*

*Child3.a,此处不是对child3的主动使用,因为a在父类中,是对

*父类的主动使用,因此初始化父类

*

*/

如有错误之处,有劳各位大佬指出

你可能感兴趣的:(JVM(Java,Virtual,Machine))