一、加载
类的加载指的是将类的.class文件中二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构。在这个阶段,会执行类中声明的静态代码块。也就是类中的静态块执行时不需要等到类的初始化。
加载.class文件的方式
1、从本地系统中直接加载
2、通过网络下载.class文件
3、从zip,jar等归档文件中加载.class文件
4、从专有数据库中提取.class文件
5、将Java源文件动态编译为.class文件
类加载的最终产品是位于堆区中的class对象,Class对象封装了类在方法区内的数据结构,并向Java程序员提供了访问方法区内的数据机构的接口. 我们可以通过类名.class来获取一个类的类型的引用,通过new 类名().getClass()来获取一个实例变量的类的引用
类的加载机制
从JDK1.2开始类加载采用父亲委托机制。除了Java虚拟机自带的根类加载器以外,其余的类加载器都有且只有一个父加载器。当Java程序骑牛加载器加载某个类时,加载器会首先委托自己的父加载器去加载该类,若父加载器能加载,则由父加载器完成加载任务,否则才由自加载器去加载。
同时,所有能成功返回Class对象的引用的类加载器(包括定义类加载器,即包括定义类加载器和它下面的所有子加载器)都被称为初始类加载器。
假设loader1实际加载了Sample类,则loader1为Sample类的定义类加载器
二、连接
类加载完成后就进入了类的连接阶段,连接阶段主要分为三个过程分别是:验证,准备和解析。在连接阶段,主要是将已经读到内存的类的二进制数据合并到虚拟机的运行时环境中去。
验证
这个阶段主要目的是保证Class流的格式是正确的。主要验证的内容包括:
1、文件格式的验证
是否以0xCAFEBABE开头
版本号是否合理2、元数据的验证
是否有父类
是否继承了final类
非抽象类实现了所有抽象方法3、字节码验证
运行检查
栈数据类型和操作码数据参数吻合
跳转指令指定到合理的位置4、符号引用验证
常量池中描述类是否存在
访问的方法或字段是否存在且有足够的权限
准备
这个阶段主要是为对象和变量分配内存,并为类设置初始值(方法区中) 对于static类型变量在这个阶段会为其赋值为默认值,比如public static int v=5,在这个阶段会为其赋值为v=0,而对于static final类型的变量,在准备阶段就会被赋值为正确的值
解析
在这个阶段会将符号引用转换成直接引用。 原来的符号引用仅仅是一个字符串,而引用的对象不一定被加载,直接引用只的是将引用对象的指针或者地址偏移量指向真正的对象,将字符串所指向的对象加载到内存中。
三、初始化
在这个阶段主要执行类的构造方法。并且为静态变量赋值为初始值,执行静态块代码。
类的初始化步骤
假如这个类还没有被加载和连接,那就先进行加载和连接
假如类存在直接的父类,并且这个父类还没有被初始化,那就先初始化它的父类
假如类中存在初始化语句时,那就依次执行这些初始化语句。
类的初始化时机
所有的Java类只有在对类的首次主动使用时才会被初始化。 主动使用的情况有六中,其他情况都属于被动使用:
1、 创建类的实例
2、访问某个类或接口的静态变量,或者对该静态变量赋值
3、调用类的静态方法
4、反射(Class.fotName)
5、初始化一个类的子类
6、Java虚拟机启动时被标明为启动类的类(面方法所在的类)
注意:
1、当Java虚拟机初始化一个类时,要求他的所有父类都已经被初始化,但是这条规则并不适合接口。在初始化一个类或接口时,并不会先初始化它所实现的接口。
2、只有当程序访问的静态变量或静态方法确实在当前类或当前接口中定义时,才可以认为是对类或接口的主动使用。如果静态方法或变量在parent中定义,从子类进行调用,则不会初始化子类。