JVM(二)——JAVA类加载过程

一、类的加载过程

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>初始化:为类中的静态变量赋予正确的初始值。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值