Java 类加载流程 双亲委派模型

2024-06-04 03:20

本文主要是介绍Java 类加载流程 双亲委派模型,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在 Java 中,类的加载是通过类加载器完成的,这个过程包括加载、连接(验证、准备、解析)和初始化三个主要阶段。

加载(Loading)

  • 读取数据:这是类加载的第一步,类加载器从文件系统、网络或其他源读取.class文件的二进制数据。
  • 生成Class对象:加载器将.class文件的二进制数据转换为java.lang.Class对象。此对象在JVM中代表这个类。

连接(Linking)

连接阶段分为验证、准备和解析三个步骤:

  • 验证(Verification):验证确保被加载的类符合JVM规范,没有安全问题。验证器检查字节码以确保它遵循Java语言的所有规则(例如,不允许使用跳转到无效位置的指令)。

  • 准备(Preparation):在准备阶段,JVM为类变量分配内存,并设置默认初始值,这些变量是被static修饰的变量。

  • 解析(Resolution):解析是将类、接口、字段和方法的符号引用转换为直接引用的过程。这意味着JVM会找出这些引用对应的实际地址。

初始化(Initialization)

  • 执行静态代码块和静态字段初始化:初始化是类加载的最后一个阶段,这个阶段中,JVM负责执行类构造器<clinit>()方法的过程。该方法是由编译器自动收集类中所有静态变量的赋值动作和静态代码块中的语句合并产生的。执行顺序是按照静态变量声明和静态代码块在代码中出现的顺序。

类加载器

Java使用类加载器来实现类的加载。主要有以下几种类型的类加载器:

  • 引导类加载器(Bootstrap Class Loader):它加载Java的核心库(JAVA_HOME/jre/lib/rt.jar等)。这个加载器是用原生代码实现的。

  • 扩展类加载器(Extension Class Loader):它加载从标准Java类库扩展的类,这些类位于JAVA_HOME/jre/lib/ext目录或者由系统属性java.ext.dirs指定的目录。

  • 系统类加载器(System Class Loader):它根据Java应用的类路径(CLASSPATH)来加载Java类。一般来说,这个加载器是我们在程序中经常使用的。

类加载器之间存在父子关系,通常使用双亲委派模型来加载类。这种模型要求除了最顶层的启动类加载器外,其它的类加载器都应有自己的父类加载器。类加载请求首先由父类加载器处理,只有当父类加载器无法满足时,才由子类加载器自己去加载。

这个过程保证了Java应用的稳定运行,也避免了类的重复加载,同时核心Java类库的类总是被引导类加载器加载,这样就避免了被恶意代码替换或篡改。

初始化阶段是线程安全的,可以以此为原理实现单例模式。

双亲委派模型的工作原理

  1. 委派父类加载器:当一个类加载器尝试加载某个类时,它首先不会自己去加载这个类,而是把类加载的请求委派给它的父类加载器去完成。如果这个类加载器有自己的父类加载器,那么递归地进行这一委派过程,直到顶层的启动类加载器(Bootstrap ClassLoader)。

  2. 顶层加载尝试:从启动类加载器开始,每层类加载器检查它是否能够加载这个类(是否已经加载过这个类),如果能够加载,就结束加载过程,返回类的Class对象;如果不能加载,委派的请求就会传递回子类加载器。

  3. 子类加载器尝试:如果所有的父类加载器都不能加载该类,最终这个加载任务会回到最初的发起者,由它来尝试加载这个类。

在特定的场合,比如在一些 Java 应用服务器中或者复杂的应用中,可能需要破坏双亲委托模型来满足特定的需求,如实现类的热替换(hot swapping)、多版本共存等。

自定义类加载器可以选择不遵守双亲委托模型。在 Java 中,可以通过继承 ClassLoader 类并重写 loadClass(String name) 方法来实现。在这个方法中,可以先尝试加载需要的类,如果失败再调用 super.loadClass(name) 来委托给父类加载器。

public class CustomClassLoader extends ClassLoader {@Overridepublic Class<?> loadClass(String name) throws ClassNotFoundException {try {// 尝试自己加载类byte[] bytes = loadClassData(name);return defineClass(name, bytes, 0, bytes.length);} catch (Exception e) {// 如果失败,委托给父类加载器return super.loadClass(name);}}private byte[] loadClassData(String name) {// 实现从文件系统、网络或其他来源加载类的字节码return ...;}
}

这篇关于Java 类加载流程 双亲委派模型的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1028925

相关文章

Spring Bean初始化及@PostConstruc执行顺序示例详解

《SpringBean初始化及@PostConstruc执行顺序示例详解》本文给大家介绍SpringBean初始化及@PostConstruc执行顺序,本文通过实例代码给大家介绍的非常详细,对大家的... 目录1. Bean初始化执行顺序2. 成员变量初始化顺序2.1 普通Java类(非Spring环境)(

Spring Boot 中的默认异常处理机制及执行流程

《SpringBoot中的默认异常处理机制及执行流程》SpringBoot内置BasicErrorController,自动处理异常并生成HTML/JSON响应,支持自定义错误路径、配置及扩展,如... 目录Spring Boot 异常处理机制详解默认错误页面功能自动异常转换机制错误属性配置选项默认错误处理

java如何实现高并发场景下三级缓存的数据一致性

《java如何实现高并发场景下三级缓存的数据一致性》这篇文章主要为大家详细介绍了java如何实现高并发场景下三级缓存的数据一致性,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 下面代码是一个使用Java和Redisson实现的三级缓存服务,主要功能包括:1.缓存结构:本地缓存:使

Java Spring的依赖注入理解及@Autowired用法示例详解

《JavaSpring的依赖注入理解及@Autowired用法示例详解》文章介绍了Spring依赖注入(DI)的概念、三种实现方式(构造器、Setter、字段注入),区分了@Autowired(注入... 目录一、什么是依赖注入(DI)?1. 定义2. 举个例子二、依赖注入的几种方式1. 构造器注入(Con

SpringBoot 异常处理/自定义格式校验的问题实例详解

《SpringBoot异常处理/自定义格式校验的问题实例详解》文章探讨SpringBoot中自定义注解校验问题,区分参数级与类级约束触发的异常类型,建议通过@RestControllerAdvice... 目录1. 问题简要描述2. 异常触发1) 参数级别约束2) 类级别约束3. 异常处理1) 字段级别约束

如何在Java Spring实现异步执行(详细篇)

《如何在JavaSpring实现异步执行(详细篇)》Spring框架通过@Async、Executor等实现异步执行,提升系统性能与响应速度,支持自定义线程池管理并发,本文给大家介绍如何在Sprin... 目录前言1. 使用 @Async 实现异步执行1.1 启用异步执行支持1.2 创建异步方法1.3 调用

java内存泄漏排查过程及解决

《java内存泄漏排查过程及解决》公司某服务内存持续增长,疑似内存泄漏,未触发OOM,排查方法包括检查JVM配置、分析GC执行状态、导出堆内存快照并用IDEAProfiler工具定位大对象及代码... 目录内存泄漏内存问题排查1.查看JVM内存配置2.分析gc是否正常执行3.导出 dump 各种工具分析4.

Spring Boot配置和使用两个数据源的实现步骤

《SpringBoot配置和使用两个数据源的实现步骤》本文详解SpringBoot配置双数据源方法,包含配置文件设置、Bean创建、事务管理器配置及@Qualifier注解使用,强调主数据源标记、代... 目录Spring Boot配置和使用两个数据源技术背景实现步骤1. 配置数据源信息2. 创建数据源Be

Spring Boot 3.x 中 WebClient 示例详解析

《SpringBoot3.x中WebClient示例详解析》SpringBoot3.x中WebClient是响应式HTTP客户端,替代RestTemplate,支持异步非阻塞请求,涵盖GET... 目录Spring Boot 3.x 中 WebClient 全面详解及示例1. WebClient 简介2.

Java中使用 @Builder 注解的简单示例

《Java中使用@Builder注解的简单示例》@Builder简化构建但存在复杂性,需配合其他注解,导致可变性、抽象类型处理难题,链式编程非最佳实践,适合长期对象,避免与@Data混用,改用@G... 目录一、案例二、不足之处大多数同学使用 @Builder 无非就是为了链式编程,然而 @Builder