一、类的加载过程
JVM将类加载过程分为3个步骤:装载(lode)、链接(link)和初始化(initialize)。链接又分为三个步骤:
(1)装载:
查找并加载类的二进制数据。
(2)链接:
1、验证:确保被加载类的正确性。。
2、准备:为类的静态变量分配内存,并将其初始化为默认值。
3、解析:把类中的符号引用转换为直接引用。
(*问题)为什么需要验证?
首先如果由编译器生成的class文件,肯定是符合JVM字节码格式的。但是万一有高手自己写了一个class文件让JVM加载并运行,用于恶意用途。因此这个class文件要先进行验证,不符合的话不会让他继续执行,也是安全考虑。
(3)初始化:
为类的静态变量赋予正确的初始值。
(*问题)准备和初始化?
准备阶段和初始化阶段看似有点矛盾,其实是不矛盾的。例如:
private static iny a = 10;
执行过程如下:首先字节码文件被加载到内存后先进性链接的验证这一步骤。验证通过后进入准备阶段,给a分配内存,因为变量a是static的,所以此时a等于int类型的默认初始值0,即a = 0。然后到解析。到初始化这一步骤时,才把a的真正值10赋给a,此时a = 10。
二、类的初始化
(1)类什么时候被初始化?
(1)创建类的实例,也就是new一个对象。
(2)访问或者某个类或接口的静态变量,或者对该静态变量进行赋值。
(3)调用类的静态方法。
(4)反射(Class.forName())
(5)初始化一个类的子类(先进行父类的初始化)。
(6)JVM启动时标明的启动类,即文件名和类名相同的那个类。
(2)类的初始化步骤?
(1)如果这个类还没有被加载和链接,先进性加载和链接。
(2)加入这个类存在父类,并且这个类还没有被初始化(在一个类加载器中类只能初始化一次)。就初始化直接地父类(不适用于接口)。
(3)加入类中存在的初始化语句(如static变量和static块),依次执行这些初始化语句。
三、类的加载
类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据的方法区内,然后再堆区穿件一个这个类的java.lang.Class对象,用来封装类在方法区类的对象。
类的加载的最终产品是位于堆区中的Class对象。Class对象封装了类在方法区内的数据结构,并且向程序员提供了调用访问方法区内数据结构的接口。
(*问题)加载类的方式?
(1)从本地系统直接加载。
(2)通过网络下载.class文件。
(3)从zip,jar等归档文件中加载.class文件。
(4)从专有数据库中提取.class文件。
(5)将java源文件动态编译为.class文件(服务器)。
四、加载器
JVM的类加载器是通过ClassLoader及其子类来完成的,
(1)类的层次关系和加载顺序如下:
1、类加载子系统(Class Loader Subsystem)
2、内存空间(运行时数据区域,Runtime Data Areas)
3、执行引擎(Execution Engine)
(2)类加载的时机
虚拟机规范中明确了在6种情况下会对类进行加载:
1、创建对象实例:new对象的时候会对类进行初始化,前提这个类没有被初始化。
2、通过class文件反射创建对象。
3、调用类的静态属性或静态属性赋值。
4、调用类的静态方法。
5、初始化一个类的子类,使用子类的时候先初始化父类。
6、Java虚拟机启动时被标记为启动类的类,比如main()方法所在的类。
(3)哪些情况下类不会被加载:
1、在同一个虚拟机中一个类只能被加载一次,如果已经被初始化的类不会被加载。
2、在编译时能确定下来的静态变量,不会对类进行初始化。
(4)JVM中类加载器的执行(双亲委派机制):
1、类加载器
Java提供了3个默认的类加载器:
<1>Bootstrap ClassLoader:负责加载特定的类($JAVA_HOME中jre/lib/rt.jar里所有的类)。
<2>Extension ClassLoader:负责加载java平台上扩展功能的jar包,包含$JAVA_HOME中jre/lib/*.jar或者-Java.ext.dirs指定路径下的jar包。
<3>Application ClassLoader:负责加载classpath中指定到jar和目录中class。
Custom ClassLoader:自定义的类加载器。
2、双亲委派
<1>当Application ClassLoader加载一个class时,它不会自己尝试加载这个类。而是将类加载请求委托给父类加载器Extension ClassLoader去完成。
<2>当Extension ClassLoader加载class时,它不会自己尝试加载这个类。而是将类加载请求委托给父类加载器Bootstrap ClassLoader去完成。
<3>如果Bootstrap ClassLoader加载失败,会使用Extension ClassLoader尝试加载。
<4>如果Extension ClassLoader加载失败,会使用Application ClassLoader尝试加载。
<5>如果ApplicationClassLoader加载失败,会抛出ClassNotFountException。
(*注)只要以上一个类加载器加载成功,就会直接返回该类。
3、好处
<1>防止内存中出现多份同样的字节码(不同类加载器指定的目录下才起作用)。
<2>单一性机制。因为委派机制的关系,一个类(唯一的全限定名)只能被一个类加载器加载。
4、加载过程
<1>加载:查找并加载类的.class文件(双亲委派机制)。
<2>验证:文件格式、元数据、字节码、符号引用。
<3>准备:为类中静态变量分配空间、并将其初始化为默认值。
<4>解析:将类中符号转化为直接引用。
<5>初始化:为类中的静态变量赋予正确的初始值。