JVM一:类加载器子系统

类加载器子系统(ClassLoader):将class字节码文件加载到JVM内存中,是否会执行该字节码文件由 执行引擎决定。

类加载器子系统(Class loader subSystem)只负责将class文件加载进内存中,并且类在仅需要时加载,并且只加载一次。至于该Class文件是否执行由 执行引擎 Execution Engine决定的

 

官方:类加载器子系统负责从文件系统或者网络中加载Class文件, 文件在文件开头有特定class的文件标识。 该特定标识指什么?CA FE BA BE 咖啡宝贝 魔术  (根据指定的起始标识可以提高安全性)

class 字节码文件的加载过程: 包含三个阶段:加载阶段-----链接阶段------初始化阶段

加载的类信息存放于在一块称为方法区的内存空间

  了解部分:

   除了类的信息外,方法区中还会存放运行时常量池信息,可能还包括字符串字面量和数字常量(这部分常量信息是Class文件中常量池部分的内存映射)

(java文件经过编译会形成class文件,class文件中有明确的结构划分,比如常量池constant pool ,当常量池加载到内存中后就被称为运行时常量池)

类加载器子系统的工作过程:

 

简述加载过程:开始---->装载HelloLoader.class文件了吗?

1. 未装载:进行装载 顺利装载执行   2 否则报错

2.成功装载 ---进行链接阶段:验证-->准备-->解析 最后初始化

类的加载过程详解:

加载模块:以文件流的形式进行加载,主要目的:生成一个代表这个类的java.lang.Class对象

1.通过一个类的全限定名获取定义此类的二进制字节流;

2.将这个字节流中所代表的静态存储结构转化为方法区的运行时数据;

3.在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问入口 方法区:会存放类信息 .class 以及静态常量;

加载class文件的方式

  • 从本地系统中直接加载
  • 通过网络获取,典型场景:Web Applet
  • 从zip压缩包中读取,成为日后jar、war格式的基础
  • 运行时计算生成,使用最多的是:动态代理技术
  • 由其他文件生成,典型场景:JSP应用从专有数据库中提取.class文件,比较少见
  • 从加密文件中获取,典型的防Class文件被反编译的保护措施

类加载器的分类

JVM支持两种类型的类加载器 。分别为引导类加载器(Bootstrap ClassLoader)和自定义类加载器(User-Defined ClassLoader 派生于抽象类ClassLoader的类)。

从概念上来讲,自定义类加载器一般指的是程序中由开发人员自定义的一类类加载器,但是Java虚拟机规范却没有这么定义,而是将所有派生于抽象类ClassLoader的类加载器都划分为自定义类加载器。

无论类加载器的类型如何划分,在程序中我们最常见的类加载器始终只有3个,如下所示:

这里的四者之间是包含关系,不是上层和下层,也不是子系统的继承关系。

类加载器的详解:

启动类加载器(引导类加载器,Bootstrap ClassLoader)

  • 这个类加载使用C/C++语言实现的,嵌套在JVM内部。rt.jar 可以理解为runtime.jar 运行时所需的jar包 比如它包含的java.io java.slq java.math ,java.lang java.time 等等我们经常使用的jar包
  • 它用来加载Java的核心库(JAVAHOME/jre/lib/rt.jar、resources.jar或sun.boot.class.path路径下的内容),用于提供JVM自身需要的类。

  • 并不继承自java.lang.ClassLoader,没有父加载器。
  • 加载扩展类和应用程序类加载器,并指定为他们的父类加载器。
  • 出于安全考虑,Bootstrap启动类加载器只加载包名为java、javax、sun等开头的类

扩展类加载器(Extension ClassLoader)

  • Java语言编写,由sun.misc.Launcher$ExtClassLoader实现。
  • 派生于ClassLoader类
  • 父类加载器为启动类加载器
  • 从java.ext.dirs系统属性所指定的目录中加载类库,或从JDK的安装目录的jre/lib/ext子目录(扩展目录)下加载类库。如果用户创建的JAR放在此目录下,也会自动由扩展类加载器加载。

应用程序类加载器(系统类加载器,AppClassLoader)

  • javI语言编写,由sun.misc.LaunchersAppClassLoader实现
  • 派生于ClassLoader类
  • 父类加载器为扩展类加载器 (ExtClassLoader)
  • 它负责加载环境变量classpath或系统属性java.class.path指定路径下的类库
  • 该类加载是程序中默认的类加载器,一般来说,Java应用的类都是由它来完成加载
  • 通过classLoader.getSystemclassLoader()方法可以获取到该类加载器

用户自定义类加载器

在Java的日常应用程序开发中,类的加载几乎是由上述3种类加载器相互配合执行的,在必要时,我们还可以自定义类加载器,来定制类的加载方式。 为什么要自定义类加载器?

  • 隔离加载类
  • 修改类加载的方式
  • 扩展加载源
  • 防止源码泄漏

用户自定义类加载器实现步骤:

  • 开发人员可以通过继承抽象类ava.lang.ClassLoader类的方式,实现自己的类加载器,以满足一些特殊的需求
  • 在JDK1.2之前,在自定义类加载器时,总会去继承ClassLoader类并重写loadClass()方法,从而实现自定义的类加载类,但是在JDK1.2之后已不再建议用户去覆盖loadclass()方法,而是建议把自定义的类加载逻辑写在findclass()方法中
  • 在编写自定义类加载器时,如果没有太过于复杂的需求,可以直接继承URIClassLoader类,这样就可以避免自己去编写findclass()方法及其获取字节码流的方式,使自定义类加载器编写更加简洁。

ClassLoader类,它是一个抽象类,自定义的类加载器都继承自ClassLoader(不包括启动类加载器)

链接模块分为三个步骤:即验证、准备、解析

  验证 verify 目的:保证被加载类的正确性

1.目的在于确保Class文件的字节流中包含信息符合当前虚拟机要求,保证被加载类的正确性,不会危害虚拟机自身安全。

2.主要包括四种验证:文件格式验证,源数据验证,字节码验证,符号引用验证。比如校字节码文件的开始标识是否正确

如果文件不合法-----------》报错提示:error 验证阶段可以有效避免 恶意撰改代码

准备 prepare 类相关信息的初始化

1.为类变量分配内存并且设置该类变量的默认初始值,即零值;

public class HelloApp { private static int a = 1; // 准备阶段为0,在下个阶段,也就是初始化的时候才是1 public static void main(String[] args) { System.out.println(a); } }

2.这里不包含用final修饰的static,因为final在编译的时候就会分配了,准备阶段会显式初始化;

3.这里不会为实例变量分配初始化,类变量会分配在方法区中,而实例变量是会随着对象一起分配到Java堆中。

解析 resolve

1.将常量池内的符号引用转换为直接引用的过程。

2.事实上,解析操作往往会伴随着jvm在执行完初始化之后再执行。

3.符号引用就是一组符号来描述所引用的目标。符号应用的字面量形式明确定义在《java虚拟机规范》的class文件格式中。直接引用就是直接指向目标的指针、相对偏移量或一个间接定位到目标的句柄

4.解析动作主要针对类或接口、字段、类方法、接口方法、方法类型等。对应常量池中的CONSTANT_Class_info/CONSTANT_Fieldref_info、CONSTANT_Methodref_info等。

初始化阶段

初始化阶段就是执行类构造器()的过程。clinit ----------->class initilazation 只会执行一次加载方法()

此方法不需定义,是javac编译器自动收集类中的所有类变量的赋值动作和静态代码块中的语句合并而来。

  • 也就是说,当我们代码中包含static变量的时候,就会有clinit方法

public class ClassInitTest { private static int num = 1; static { num = 2; number = 20; System.out.println(num); System.out.println(number); //报错,非法的前向引用 } private static int number = 10; public static void main(String[] args) { System.out.println(ClassInitTest.num); // 2 System.out.println(ClassInitTest.number); // 10 } }

()方法中指令按语句在源文件中出现的顺序执行。

()不同于类的构造器。(关联:构造器是虚拟机视角下的())若该类具有父类,JVM会保证子类的()执行前,父类的()已经执行完毕。

  • 任何一个类在声明后,都有生成一个构造器,默认是空参构造器
public class ClinitTest1 { static class Father { public static int A = 1; static { A = 2; } } static class Son extends Father { public static int b = A; } public static void main(String[] args) { System.out.println(Son.b);//2 } }

我们输出结果为 2,也就是说首先加载ClinitTest1的时候,会找到main方法,然后执行Son的初始化,但是Son继承了Father,因此还需要执行Father的初始化,同时将A赋值为2。我们通过反编译得到Father的加载过程,首先我们看到原来的值被赋值成1,然后又被复制成2,最后返回

iconst_1 putstatic #2 <com/atguigu/java/chapter02/ClinitTest1$Father.A> iconst_2 putstatic #2 <com/atguigu/java/chapter02/ClinitTest1$Father.A> return

虚拟机必须保证一个类的Clinit()方法在多线程下被同步加锁。

public class DeadThreadTest { public static void main(String[] args) { new Thread(() -> { System.out.println(Thread.currentThread().getName() + "\t 线程t1开始"); new DeadThread(); }, "t1").start(); new Thread(() -> { System.out.println(Thread.currentThread().getName() + "\t 线程t2开始"); new DeadThread(); }, "t2").start(); } } class DeadThread { static { if (true) { System.out.println(Thread.currentThread().getName() + "\t 初始化当前类"); while(true) { } } } }

上面的代码,输出结果为

线程t1开始 线程t2开始 线程t2 初始化当前类

从上面可以看出初始化后,只能够执行一次初始化,这也就是同步加锁的过程

30 集需要在打卡

======================================================================================================

双亲委派机制(重点)

Java虚拟机对class文件采用的是按需加载的方式(可以类比懒加载)也就是说:当需要使用到该类时才会将它的class文件加载到内存生成class对象。二而且加载某个l类的class文件是,Java虚拟机采用的是双亲委派模式,即把加载class文件的请求交由父类处理,它是一种任务委派模式。

工作原理:

1.如果一个类加载器收到了类加载请求,它并不会自己先去加载,而是将这个加载请求委托给父类的加载器去执行;

2.如果父类加载器还存在其父类加载器,则会进一步向上委托,依次递归,请求最终会到达顶层的启动类加载器BootstrapClassLoader;

3.如果父类加载器可以完成类加载任务,就成功返回。倘若父类加载器不能够正常加载,才会交由子类的类加载器进行加载,直到将该类加载成功。注意:一个class类只会加载一次(它相当于模板)这就是双亲委派机制。

双亲委派机制的优势:

避免类Class的重复加载,(由双亲委派机制的原理所知:一个class类只会加载一次(它相当于模板))

保护程序安全,防止核心API被随意篡改

  • 自定义类:java.lang.String (由于java.lang包是位于/jre/lib/rt.jar下,由BootStrapClassLoader 类加载器进行加载的,保证了核心API不会被错误加载 )
  • 自定义类:java.lang.ShkStart(报错:阻止创建 java.lang开头的类)

如何判断两个class对象是否相同:

在JVM中表示两个class对象是否为同一个类存在两个必要条件:

  • 类的完整类名必须一致,包括包名。 (全类名必须一致)
  • 加载这个类的ClassLoader(指ClassLoader实例对象)必须相同。 (加载的ClassLoader必须一致)

比如 在java.lang包下存在 String.java 和string.java(自定义),虽然一致 来源同一个Class文件,但一个是由BootStrapClassLoader 加载、另一个是由AppClassLoader加载,则 :这两个类对象也是不相等的。

JVM必须知道一个类型是由启动加载器加载的还是由用户类加载器加载的。如果一个类型是由用户类加载器加载的,那么JVM会将这个类加载器的一个引用作为类型信息的一部分保存在方法区中。

当解析一个类型到另一个类型的引用的时候,JVM需要保证这两个类型的类加载器是相同的。

类的主动使用和被动使用

Java程序对类的使用方式分为:主动使用和被动使用。 主动使用,又分为七种情况:

  • 创建类的实例
  • 访问某个类或接口的静态变量,或者对该静态变量赋值
  • 调用类的静态方法I
  • 反射(比如:Class.forName("com.atguigu.Test"))
  • 初始化一个类的子类
  • Java虚拟机启动时被标明为启动类的类
  • JDK7开始提供的动态语言支持:
  • java.lang.invoke.MethodHandle实例的解析结果REF getStatic、REF putStatic、REF invokeStatic句柄对应的类没有初始化,则初始化

除了以上七种情况,其他使用Java类的方式都被看作是对类的被动使用,都不会导致类的初始化。(注意)

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值