JVM 类加载机制

目录

一、类加载过程

[1.1 加载](#1.1 加载)

[1.2 连接](#1.2 连接)

[1.3 初始化](#1.3 初始化)

二、双亲委派模式

[2.1 什么是双亲委派模式](#2.1 什么是双亲委派模式)

[2.2 双亲委派模式的优点](#2.2 双亲委派模式的优点)

[2.3 破坏双亲委派模型](#2.3 破坏双亲委派模型)


一、类加载过程

对于一个类的生命周期来说:


其中前 5 步是固定的顺序并且也是类加载的过程,其中中间的 3 步都属于连接,所以对于类加载

来说总共分为以下几个步骤:

  1. 加载
  2. 连接:验证、准备、解析
  3. 初始化

1.1 加载

"加载"(Loading)阶段是整个"类加载"(Class Loading)过程中的一个阶段,它和类加载 Class Loading 是不同的,一个是加载 Loading 另一个是类加载 Class Loading。
在加载 Loading 阶段,Java虚拟机需要完成以下三件事情:

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

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

  3. 在内存中生成一个代表这个类的 java.lang.Class 对象,作为方法区这个类的各种数据的访问入口。

1.2 连接

验证是连接阶段的第一步,这以阶段的目的是确保Class文件的字节流中包含的信息符合《Java虚拟机规范》的全部约束要求,保证这些信息被当作代码运行后不会危害虚拟机自身的安全。

验证选项: 文件格式验证、字节码验证、符号引用验证...
准备阶段是正式为类中定义的变量(即静态变量,被static修饰的变量)分配内存并设置类变量初始值的阶段。

比如此时有这样一行代码:

java 复制代码
public static int value = 123;

它是初始化 value 的 int 值为 0,而非 123。

解析阶段是 Java 虚拟机将常量池内的符号引用替换为直接引用的过程,也就是初始化常量的过程。

1.3 初始化

初始化阶段,Java 虚拟机真正开始执行类中编写的 Java 程序代码,将主导权移交给应用程序。初始化阶段就是执行类构造器方法的过程。

加载:在硬盘上,找到对应的.class文件,读取文件内容。

验证:检查.class里的内容,是否符合要求。

准备:给类对象分配内存空间。

解析:针对字符串常量来初始化。把刚才.class文件中的常量内容取出来,放到"元数据区"。

初始化:针对类对象进行初始化。

二、双亲委派模式

2.1 什么是双亲委派模式

如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次的类加载器都是如此,因此所有的加载请求最终都应该传送到最顶层的启动类加载器中,只有当父加载器反馈自己无法完成这个加载请求(它的搜索范围中没有找到所需的类)时,子加载器才会尝试自己去完成加载。

  • 启动类加载器:加载 JDK 中 lib目录中 Java 的核心类库,即$JAVA_HOME/lib目录。 扩展类加载器。加载 lib/ext目录下的类。

  • 应用程序类加载器:加载我们写的应用程序。

  • 自定义类加载器:根据自己的需求定制类加载器。

2.2 双亲委派模式的优点

  1. 避免重复加载类:比如 A 类和 B 类都有一个父类 C 类,那么当 A 启动时就会将 C 类加载起来,那么在 B 类进行加载时就不需要在重复加载 C 类了。

  2. 安全性:使用双亲委派模型也可以保证了 Java 的核心 API 不被篡改,如果没有使用双亲委派模型,而是每个类加载器加载自己的话就会出现一些问题,比如我们编写一个称为 java.lang.Object 类的话,那么程序运行的时候,系统就会出现多个不同的 Object 类,而有些 Object 类又是用户自己提供的因此安全性就不能得到保证了。

2.3 破坏双亲委派模型

双亲委派模型虽然有其优点,但在某些情况下也存在一定的问题,比如 Java 中 SPI(Service Provider Interface,服务提供接口)机制中的 JDBC 实现。
JDBC 的 Driver 接口定义在 JDK 中,其实现由各个数据库的服务商来提供,比如 MySQL 驱动包。 JDBC 的核心使用代码:

java 复制代码
public class JdbcTest {
 public static void main(String[] args){
 Connection connection = null;
 try {
 connection = DriverManager.getConnection("jdbc:mysql://127.0.0.1:330
 } catch (SQLException e) {
 e.printStackTrace();
 }
 System.out.println(connection.getClass().getClassLoader());
 System.out.println(Thread.currentThread().getContextClassLoader());
 System.out.println(Connection.class.getClassLoader());
 }
}

进入 DriverManager 的源码类就会发现它是存在系统的 rt.jar 中的。

由双亲委派模型的加载流程可知 rt.jar 是有顶级父类 Bootstrap ClassLoader 加载的,如下图所示:

而当我们进入它的 getConnection 源码是却发现,它在调用具体的类实现时,使用的是子类加载器 (线程上下文加载器 Thread.currentThread().getContextClassLoader )来加载具体的数据库数据库包(如 mysql 的 jar 包),源码如下:

java 复制代码
 @CallerSensitive
 public static Connection getConnection(String url,
 java.util.Properties info) throws SQLException {
 return (getConnection(url, info, Reflection.getCallerClass()));
 }
 private static Connection getConnection(
 String url, java.util.Properties info, Class<?> caller) throws SQLExcept
 ClassLoader callerCL = caller != null ? caller.getClassLoader() : null;
 synchronized(DriverManager.class) {
 // synchronize loading of the correct classloader.
 if (callerCL == null) {
 //获取线程上下为类加载器
 callerCL = Thread.currentThread().getContextClassLoader();
 }
 }

 if(url == null) {
throw new SQLException("The url cannot be null", "08001");
 }
 println("DriverManager.getConnection(\"" + url + "\")");
 SQLException reason = null;
 for(DriverInfo aDriver : registeredDrivers) {
 // isDriverAllowed 对于 mysql 连接 jar 进⾏加载
 if(isDriverAllowed(aDriver.driver, callerCL)) {
 try {
 println(" trying " + aDriver.driver.getClass().getName())
 Connection con = aDriver.driver.connect(url, info);
 if (con != null) {
 // Success!
 println("getConnection returning " + aDriver.driver.getC
 return (con);
 }
 } catch (SQLException ex) {
 if (reason == null) {
 reason = ex;
 }
 }
 } else {
 println(" skipping: " + aDriver.getClass().getName());
 }
 }
 if (reason != null) {
 println("getConnection failed: " + reason);
 throw reason;
 }
 println("getConnection: no suitable driver found for "+ url);
 throw new SQLException("No suitable driver found for "+ url, "08001");
 }

这样一来就破坏了双亲委派模型,因为 DriverManager 位于 rt.jar 包,由 BootStrap 类加载器加载, 而其 Driver 接口的实现类是位于服务商提供的 Jar 包中,是由子类加载器(线程上下文加载器Thread.currentThread().getContextClassLoader )来加载的,这样就破坏了双亲委派模型了(双亲委派模型讲的是所有类都应该交给父类来加载,但 JDBC 显然并不能这样实现)。它的交互流程图如下所示:

相关推荐
Yeats_Liao4 分钟前
Spring 框架:配置缓存管理器、注解参数与过期时间
java·spring·缓存
Yeats_Liao4 分钟前
Spring 定时任务:@Scheduled 注解四大参数解析
android·java·spring
码明4 分钟前
SpringBoot整合ssm——图书管理系统
java·spring boot·spring
某风吾起8 分钟前
Linux 消息队列的使用方法
java·linux·运维
xiao-xiang11 分钟前
jenkins-k8s pod方式动态生成slave节点
java·kubernetes·jenkins
网络风云13 分钟前
golang中的包管理-下--详解
开发语言·后端·golang
取址执行23 分钟前
Redis发布订阅
java·redis·bootstrap
小唐C++31 分钟前
C++小病毒-1.0勒索
开发语言·c++·vscode·python·算法·c#·编辑器
S-X-S36 分钟前
集成Sleuth实现链路追踪
java·开发语言·链路追踪
快乐就好ya1 小时前
xxl-job分布式定时任务
java·分布式·spring cloud·springboot