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

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

Security OAuth2 单点登录流程

单点登录(英语:Single sign-on,缩写为 SSO),又译为单一签入,一种对于许多相互关连,但是又是各自独立的软件系统,提供访问控制的属性。当拥有这项属性时,当用户登录时,就可以获取所有系统的访问权限,不用对每个单一系统都逐一登录。这项功能通常是以轻型目录访问协议(LDAP)来实现,在服务器上会将用户信息存储到LDAP数据库中。相同的,单一注销(single sign-off)就是指

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

大模型研发全揭秘:客服工单数据标注的完整攻略

在人工智能(AI)领域,数据标注是模型训练过程中至关重要的一步。无论你是新手还是有经验的从业者,掌握数据标注的技术细节和常见问题的解决方案都能为你的AI项目增添不少价值。在电信运营商的客服系统中,工单数据是客户问题和解决方案的重要记录。通过对这些工单数据进行有效标注,不仅能够帮助提升客服自动化系统的智能化水平,还能优化客户服务流程,提高客户满意度。本文将详细介绍如何在电信运营商客服工单的背景下进行

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory