添加链接
link管理
链接快照平台
  • 输入网页链接,自动生成快照
  • 标签化管理网页链接
## 什么是类的加载过程 一个Java文件从编码完成到最终执行,一般主要包括两个过程:编译和运行,其中编译就是把我们写好的java文件,通过javac命令编译成字节码,也就是我们常说的.class文件,然后运行则是把编译声称的.class文件交给Java虚拟机(JVM)执行。 而我们所说的类加载过程即是指JVM虚拟机把.class文件中类信息加载进内存,并进行解析生成对应的class对象的过程。 举个通俗点的例子来说,JVM在执行某段代码时,遇到了class A, 然而此时内存中并没有class A的相关信息,于是JVM就会到相应的class文件中去寻找class A的类信息,并加载进内存中,这就是我们所说的类加载过程。 由此可见,JVM不是一开始就把所有的类都加载进内存中,而是只有第一次遇到某个需要运行的类时才会加载,且只加载一次。 ## 类加载的过程讲解 类加载的过程主要分为三个部分:加载、链接、初始化,这三个阶段 ### 第一个部分:加载阶段 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个这个类的java.lang.Class对象,用来封装类在方法区类的对象。主要步骤可以分为下面的三件事情: ```shell 1、通过一个类的全限定名来获取其定义的二进制字节流。 2、将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构。 3、在Java堆中生成一个代表这个类的java.lang.Class对象,作为对方法区中这些数据的访问入口。 加载阶段完成后,虚拟机外部的 二进制字节流就按照虚拟机所需的格式存储在方法区之中,而且在Java堆中也创建一个java.lang.Class类的对象,这样便可以通过该对象访问方法区中的这些数据。 ![这里写图片描述][itqiankun.com_805452831.png] ![这里写图片描述][itqiankun.com_805452832.png] 类的加载的最终产品是位于堆区中的Class对象。Class对象封装了类在方法区内的数据结构,并且向Java程序员提供了访问方法区内的数据结构的接口。加载类的方式有以下几种: >  1)从本地系统直接加载   2)通过网络下载.class文件   3)从zip,jar等归档文件中加载.class文件   4)从专有数据库中提取.class文件   5)将Java源文件动态编译为.class文件(服务器) #### 加载器   JVM的类加载是通过ClassLoader及其子类来完成的,类的层次关系和加载顺序可以由下图来描述: ![这里写图片描述][itqiankun.com_805452833.png] 1)BootstrapClassLoader(启动类加载器)   负责加载`$JAVA_HOME中jre/lib/rt.jar`里所有的class,加载System.getProperty(“sun.boot.class.path”)所指定的路径或jar。 2)ExtensionClassLoader(标准扩展类加载器)   负责加载java平台中扩展功能的一些jar包,包括$JAVA\_HOME中jre/lib/\*.jar或-Djava.ext.dirs指定目录下的jar包。载System.getProperty(“java.ext.dirs”)所指定的路径或jar。 3)AppClassLoader(系统类加载器)   负责记载classpath中指定的jar包及目录中class 4)CustomClassLoader(自定义加载器)   属于应用程序根据自身需要自定义的ClassLoader,如tomcat、jboss都会根据j2ee规范自行实现。 (2)类加载器的顺序 1)加载过程中会先检查类是否被已加载,检查顺序是自底向上,从Custom ClassLoader到BootStrap ClassLoader逐层检查,只要某个classloader已加载就视为已加载此类,保证此类只所有ClassLoader加载一次。而加载的顺序是自顶向下,也就是由上层来逐层尝试加载此类。 2)在加载类时,每个类加载器会将加载任务上交给其父,如果其父找不到,再由自己去加载。 3)Bootstrap Loader(启动类加载器)是最顶级的类加载器了,其父加载器为null。 > 在类的加载过程中,相对于类加载过程里面的的连接和初始化阶段而言,加载阶段是可控性最强的阶段,因为开发人员既可以使用系统提供的类加载器来完成加载,也可以自定义自己的类加载器来完成加载。 >> 我记得阿里的pandora就是通过不同的类加载器来加载不同的类,可以去类加载器里面去看阿里的pandora框架知识点 ### 第二个部分:连接阶段 在连接里面又可以被分成3个小阶段,分别是:验证,准备,解析 #### 连接阶段之验证小阶段 **验证的目的** ```java 主要是为了保证加载进来的字节流符合虚拟机规范,不会造成安全错误。 1.包括对于文件格式的验证,比如常量中是否有不被支持的常量?文件中是否有不规范的或者附加的其他信息? 2.对于元数据的验证,比如该类是否继承了被final修饰的类?类中的字段,方法是否与父类冲突?是否出现了不合理的重载? 3.对于字节码的验证,保证程序语义的合理性,比如要保证类型转换的合理性。 4.对于符号引用的验证,比如校验符号引用中通过全限定名是否能够找到对应的类?校验符号引用中的访问性(private,public等)是否可被当前类访问? **验证做的具体内容** 验证是连接阶段的第一步,这一阶段的目的是为了确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。验证阶段大致会完成4个阶段的检验动作: ```java 1.文件格式验证:验证字节流是否符合Class文件格式的规范; 例如:是否以0xCAFEBABE开头、主次版本号是否在当前虚拟机的处理范围之内、常量池中的常量是否有不被支持的类型。 2.元数据验证:对字节码描述的信息进行语义分析(注意:对比javac编译阶段的语义分析), 以保证其描述的信息符合Java语言规范的要求;例如:这个类是否有父类,除了java.lang.Object之外。 3.字节码验证:通过数据流和控制流分析,确定程序语义是合法的、符合逻辑的,如循环、分支等 4.符号引用验证:确保解析动作能正确执行,比如不能访问引用类的私有方法、全限定名称是否能找到相关的类。 验证阶段是非常重要的,但不是必须的,它对程序运行期没有影响,如果所引用的类经过反复验证,那么可以考虑采用-Xverifynone参数来关闭大部分的类验证措施,以缩短虚拟机类加载的时间。 #### 连接阶段之准备小阶段 **在准备阶段,为静态变量的初值为jvm默认的初值,而不是我们在程序中设定的初值。jvm默认为静态变量的初值是这样的** ```java 1.静态变量是基本类型(int、long、short、char、byte、boolean、float、double)的默认值为0 2.静态变量时引用类型的默认值为null 3.被final和static共同修改的静态变量,我们通常称为常量,然后常量的默认值为我们程序中设定的值, 比如我们在程序中定义final static int a = 100,则准备阶段中a的初值就是100。 为什么被final和static变量修饰的成员变量在准备阶段的赋值会比较特别呢,这是因为,被final和static修改的变量,我们叫做ConstantValue属性,ConstantValue属性就是这样特殊的属性,至于什么是ConstantValue属性,看这篇文章:https://blog.csdn.net/weixin_43689480/article/details/96099841 #### 连接阶段之解析小阶段 **这一阶段的任务就是把常量池中的符号引用转换为直接引用** 什么是符号引用,什么是直接引用 ```java 符号引用:即一个字符串,但是这个字符串给出了一些能够唯一性识别一个方法,一个变量,一个类的相关信息。 直接引用:可以理解为一个内存地址,或者一个偏移量。比如类方法,类变量的直接引用是指向方法区的指针; 在解析阶段,虚拟机会把所有的类名,方法名,字段名这些符号引用替换为具体的内存地址或偏移量,也就是直接引用。 举个例子来说,现在调用方法hello(),这个方法的地址是1234567,那么hello就是符号引用,1234567就是直接引用。 ## 第三个部分:初始阶段 ### 初始化阶段工作内容: JVM负责主要对类变量(类变量就是static修改的变量)进行初始化 这里主要对类变量(类变量就是static修改的变量)进行初始化,初始化主要有两个方式 ```java 1.声明静态类变量时指定初始值 2.使用静态代码块为类变量指定初始值 ### 初始化时机 类初始化时机:只有当对类的主动使用的时候才会导致类的初始化,类的主动使用包括以下六种: ```java – 创建类的实例,也就是new的方式 – 访问某个类或接口的静态变量,或者对该静态变量赋值 – 调用类的静态方法 – 反射(如Class.forName(“com.shengsiyuan.Test”)) – 初始化某个类的子类,则其父类也会被初始化 – Java虚拟机启动时被标明为启动类的类(Java Test),直接使用java.exe命令来运行某个主类 ### 初始化顺序   1)如果这个类还没有被加载和链接,那先进行加载和链接   2)假如这个类存在直接父类,并且这个类还没有被初始化(注意:在一个类加载器中,类只能初始化一次),那就初始化直接的父类(不适用于接口)   3)加入类中存在初始化语句(如static变量和static块),那就依次执行这些初始化语句。 如果有父类,则顺序是:父类static方法/static变量赋值 –> 子类static方法/static变量赋值 [itqiankun.com_805452831.png]: https://itqiankun.oss-cn-beijing.aliyuncs.com/picture/blogArticles_new/2021/05/06/itqiankun.com_805452831.png [itqiankun.com_805452832.png]: https://itqiankun.oss-cn-beijing.aliyuncs.com/picture/blogArticles_new/2021/05/06/itqiankun.com_805452832.png [itqiankun.com_805452833.png]: https://itqiankun.oss-cn-beijing.aliyuncs.com/picture/blogArticles_new/2021/05/06/itqiankun.com_805452833.png