Java - 类加载器

article/2025/9/18 8:50:39

文章目录

  • 1. 类加载的过程
  • 2. 类加载器的分类
    • 2.1 引导类加载器 Bootstrap
    • 2.2 扩展类加载器 ExtClassLoader
    • 2.3 系统类加载器 AppClassLoader
    • 2.4 三者之间的关系
    • 2.5 自定义类加载器
    • 2.6 注:谁来准备类加载器呢?
  • 3. 双亲委派机制
  • 4. ClassLoader抽象类
  • 5. URLClassLoader类
  • 6. 参考资料

1. 类加载的过程

在这里插入图片描述

加载:将字节码文件通过IO流读取到JVM的方法区,并同时在堆中生成Class对像。

验证:校验字节码文件的正确性。

准备:为类的静态变量分配内存,并初始化为默认值;对于final static修饰的变量,在编译时就已经分配好内存了。

解析:将类中的符号引用转换为直接引用。

初始化:对类的静态变量初始化为指定的值,执行静态代码。

2. 类加载器的分类

主要分为两类:1. JVM内置的类加载器,Bootstrap加载器、ExtClassLoader加载器和AppClassLoader加载器 三种,分别负责加载不同目录下的.class文件2. 用户自定义的类加载器,负责的加载目录自己决定。

2.1 引导类加载器 Bootstrap

引导类加载器属于JVM的一部分,由C++代码实现。

引导类加载器负责加载<JAVA_HOME\>\jre\lib路径下的核心类库,由于安全考虑只加载 包名 java、javax、sun开头的类。

package classloader.bootstrap;import sun.misc.Launcher;import java.net.URL;public class Demo01 {public static void main(String[] args) {//Bootstrap 引导类加载器//打印为null,是因为Bootstrap是C++实现的。ClassLoader classLoader = Object.class.getClassLoader();System.out.println(classLoader);//查看引导类加载器会加载那些jar包URL[] urLs = Launcher.getBootstrapClassPath().getURLs();for (URL urL : urLs) {System.out.println(urL);}}
}

2.2 扩展类加载器 ExtClassLoader

全类名:sum.misc.Launch$ExtClassLoader,Java语言实现。

扩展类加载器的父加载器是Bootstrap启动类加载器 (注:不是继承关系)

扩展类加载器负责加载<JAVA_HOME>\jre\lib\ext目录下的类库。

加载的jar包
在这里插入图片描述
获取扩展类加载器
在这里插入图片描述
注: JDK9是jdk.internal.loader.ClassLoaders$PlatformClassLoader

2.3 系统类加载器 AppClassLoader

全类名: sun.misc.Launcher$AppClassLoader

系统类加载器的父加载器是ExtClassLoader扩展类加载器(注: 不是继承关系)。

系统类加载器负责加载 classpath环境变量所指定的类库,是用户自定义类的默认类加载器。

获取系统类加载器
在这里插入图片描述
注: JDK9是jdk.internal.loader.ClassLoaders$AppClassLoader

2.4 三者之间的关系

AppClassLoader的父加载器是ExtClassLoader
ExtClassLoader的父加载器是Bootstrap
Bootstrap是根加载器三者之间是没有继承关系的。
AppClassLoaderExtClassLoader都实现了抽象类ClassLoader。抽象类ClassLoader有一个字段parent, AppClassLoaderExtClassLoader通过设置该字段引用,指定父加载器。(是组合关系)AppClassLoader 的parent指向 ExtClassLoader
ExtClassLoader 的parent指向 null,(null的原因是因为BootstrapC++实现的,通过代码中逻辑判断来转向Bootstrap)// The parent class loader for delegation
// Note: VM hardcoded the offset of this field, thus all new fields
// must be added *after* it.
private final ClassLoader parent;

2.5 自定义类加载器

自定义类加载器是为了加载在jvm三个加载器负责的目录范围之外的类

package com;import java.io.*;/*** @Date: 2022/5/2 10:09* @author: ZHX* @Description: 自定义类加载器*/
public class MyClassLoader extends ClassLoader {private String classPath;public MyClassLoader(String classPath) {this.classPath = classPath;}//parent: 指定父加载器, AppClassLoader/ExtClassLoader/Bootstrappublic MyClassLoader(ClassLoader parent, String classPath) {super(parent);this.classPath = classPath;}@Overrideprotected Class<?> findClass(String name) throws ClassNotFoundException {//要求返回的是你要加载的字节码文件的Class对象.//这里都是我们说了算的。//步骤://1. 从本地或网络某处读一个输入流到内存中 .//2. 将流内容字节数组 封装成Class对象 (直接调ClassLoader的defineClass方法,JVM会帮我们按照.class文件格式创建好的。)//1.//处理得到完整路径String path = this.classPath + name.replace(".", File.separator) + ".class";//2.读取到内存try (FileInputStream fis = new FileInputStream(path); ByteArrayOutputStream baos = new ByteArrayOutputStream()) {byte[] buffer = new byte[1024];int len = 0;while ((len = fis.read(buffer)) != -1) {//用ByteArrayOutputStream暂存一下。baos.write(buffer, 0, len);}byte[] allByte = baos.toByteArray();//将字节数组生成Class对象return super.defineClass(name, allByte, 0, allByte.length);} catch (IOException e) {throw new ClassNotFoundException(name + "加载失败");}}//测试下public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException {//使用自己的类加载器,加载D:\\ + com.ali.HelloMyClassLoader myClassLoader = new MyClassLoader("d:\\");  ////加载 全限定名类Class<?> clazz = myClassLoader.loadClass("com.ali.Hello");clazz.newInstance();System.out.println(clazz.getClassLoader()); //out: 使用的类加载器 MyClassLoader@481248}
}

加载jar包的写法:

从jar包加载类:String path = "jar:file:\\" + classPath + "!/" + name.replace(".", File.separator) + ".class";

2.6 注:谁来准备类加载器呢?

AppClassLoader和ExtClassLoader是Launcher的静态内部类,在程序启动时JVM会创建Launcher对象,Launcher构造器会同时会创建扩展类加载器和应用类加载器。

Launcher类
在AAAA述

3. 双亲委派机制

双亲委派机制就是: 每个类加载器都很懒,加载类时都先让父加载器去尝试加载,父加载器加载不了时自己才去加载。

在这里插入图片描述
图片来源 : https://www.bilibili.com/video/BV16T4y1P79h?p=2

例如: 加载自定义类Demo.class的流程

  1. 首先使用AppClassLoader类加载器尝试加载,AppClassLoader加载器会先检查它的缓存,查看该类是否已经被加载,有则不加载,没有则向上交给ExtClassLoader加载器。
  2. ExtClassLoader加载器同样会先检查它的缓存,查看该类是否已经被加载,有则不加载,没有则向上交给Bootstrap加载器。
  3. Bootstrap加载器同样会先检查它的缓存,查看该类是否已经被加载。有则不加载,没有则尝试从它负责的目录中加载,
  4. Bootstrap加载器加载失败(不在它负责的目录范围)则向下交给ExtClassLoader加载器。
  5. ExtClassLoader加载器会从它负责的目录中尝试加载,加载失败则向下交给AppClassLoader加载器
  6. AppClassLoader加载器从它负责的classpath尝试加载,加载完成。

双亲委派机制的好处:

  1. 避免类的重复加载:当父加载器已经加载该类时,就没有必要子加载器再加载一遍,保证被加载类的唯一性。

  2. 同时Java有沙箱安全机制:自定义类的包名以 java.开头被禁止, 防止核心API被篡改,判断逻辑在defineClass方法中。在这里插入图片描述

打破双亲委派机制

双亲委派机制的实现其实就是在loadClass方法中实现的。
直接调用findClass方法就可以跳过双亲委派机制,这样就可以直接加载,而不用向上委托了。
public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException {//使用上面自定义的类加载器。MyClassLoader myClassLoader1 = new MyClassLoader("d:\\");//find方法调用,加载 全限定名类Class<?> clazz1 = myClassLoader1.findClass("com.ali.Hello");System.out.println(clazz1.hashCode()); //out: 26508395System.out.println(clazz1.getClassLoader()); //out: 使用的类加载器 MyClassLoader@481248
}
//如果要想一个类加载两次,就需要创建两个类加载器。(因为判断缓存中该字节码文件是否已经已经被加载是在defineClass方法中,而该方法为final我们没法改写.)
public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException {//使用上面自定义的类加载器。MyClassLoader myClassLoader1 = new MyClassLoader("d:\\");MyClassLoader myClassLoader2 = new MyClassLoader("d:\\");//加载 全限定名类Class<?> clazz1 = myClassLoader1.findClass("com.ali.Hello");Class<?> clazz2 = myClassLoader2.findClass("com.ali.Hello");System.out.println(clazz1.hashCode());//out: 22913620System.out.println(clazz2.hashCode());//out: 29768086System.out.println(clazz1.getClassLoader()); //out: 使用的类加载器 MyClassLoader@481248System.out.println(clazz2.getClassLoader()); //out: 使用的类加载器 MyClassLoader@1947c6b
}

4. ClassLoader抽象类

所有的类加载器(除了Bootstrap)都要继承ClassLoader抽象类。

主要方法

方法名作用
public Class<?> loadClass(String name)双亲委派机制的实现
protected Class<?> findClass(String name)读取字节码文件到内存并调用defindClass方法生成Class对象
protected final Class<?> defineClass(String name, byte[] b, int off, int len)先判断是否加载过,然后将字节数组解析成Class对象
protected final void resolveClass(Class<?> c)连接指定的类

loadClass()方法源码

protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {synchronized (getClassLoadingLock(name)) {// First, check if the class has already been loaded// 检查类是否已经被加载了Class<?> c = findLoadedClass(name);if (c == null) { //没有long t0 = System.nanoTime();try {//双亲委派机制加载if (parent != null) {c = parent.loadClass(name, false);} else {c = findBootstrapClassOrNull(name);}} catch (ClassNotFoundException e) {// ClassNotFoundException thrown if class not found// from the non-null parent class loader}if (c == null) {// If still not found, then invoke findClass in order// to find the class.long t1 = System.nanoTime();//真正将字节码文件加载到内存。c = findClass(name);// this is the defining class loader; record the statssun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);sun.misc.PerfCounter.getFindClasses().increment();}}if (resolve) {resolveClass(c);}return c;}
}

5. URLClassLoader类

java.net.URLClassLoader继承了ClassLoader类. 拓展了功能,能够从网络或本地加载类。默认的父加载器是AppClassLoader系统类加载器。
在这里插入图片描述
加载磁盘上的类

package classloader.urlclassloader;/*** @Date: 2022/4/30 9:35* @author: ZHX* @Description:*/
public class LoadLocal {public LoadLocal(){System.out.println("本地的字节码文件");}}
package classloader.urlclassloader;import java.io.File;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;/*** @Date: 2022/4/30 0:51* @author: ZHX* @Description:*/
public class Demo {public static void main(String[] args) throws MalformedURLException, ClassNotFoundException, InstantiationException, IllegalAccessException {File file = new File("D:\\Project\\java-advence\\day16-classloader\\src\\main\\java\\");URL url = file.toURI().toURL();URLClassLoader urlClassLoader = new URLClassLoader(new URL[]{url});Class<?> clazz = urlClassLoader.loadClass("classloader.urlclassloader.LoadLocal");clazz.newInstance(); //out: 本地的字节码文件}
}

加载网络上的类

package classloader.urlclassloader;/*** @Date: 2022/4/30 9:35* @author: ZHX* @Description:*/
public class LoadURL {public LoadURL(){System.out.println("网络上的字节码文件");}
}
package classloader.urlclassloader;import java.io.File;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;/*** @Date: 2022/4/30 0:51* @author: ZHX* @Description:*/
public class Demo {public static void main(String[] args) throws MalformedURLException, ClassNotFoundException, InstantiationException, IllegalAccessException {//放到了tomcat服务器上。URL url = new URL("http://localhost/");URLClassLoader urlClassLoader = new URLClassLoader(new URL[]{url});Class<?> clazz = urlClassLoader.loadClass("classloader.urlclassloader.LoadURL");clazz.newInstance(); //out: 网络上的字节码文件}
}

6. 参考资料

阿里P7面试题,Java程序员手写自定义类加载器,讲透Java类加载全过程

面试官:谈谈类加载器吧,你有没有看过类加载器的源码?(总结的十分详细)

系统学习让你轻松定义java类加载器

AppClassLoader

类的加载与类加载器


http://chatgpt.dhexx.cn/article/yoD8Bl8T.shtml

相关文章

DAU 和 MAU

DAU 和 MAU 日活跃用户占月活跃用户的比例越高&#xff0c;表明用户对App的使用粘性越高。 DAU&#xff0c;即&#xff1a;Daily Active User&#xff0c;指日活跃用户数 MAU&#xff0c;即&#xff1a;Monthly Active User&#xff0c;指月活跃用户数。 *例子1&#xff1a;…

如何通过DAU分析活跃用户?(案例:python绘制箱体图)

前言&#xff1a;本文内容以游戏产品为基础进行讲解&#xff0c;内容为以下4部分&#xff1a; 1. 如何理解DAU反映了哪些问题&#xff1f; 2. 有哪些因素会影响DAU变动&#xff1f; 3. 如何解读DAU的“箱体图”&#xff1f; 4. 如何使用python绘制“箱体图”&#xff1f; DAU的…

【数据分析】产品日活DAU下降,怎么分析

目录 案例简介 第一步&#xff1a;确认数据真实性 第二步&#xff1a;明确定义&#xff0c;并拆解指标&#xff0c;进一步定位原异常部分 第三步&#xff1a;根据几个常见维度初步拆分数据 第四步&#xff1a;进一步做假设并细分深入&#xff0c;得出结论 案例分析 例题…

数据分析——DAU下降问题(转)

文章转自&#xff1a; DAU异常下降该如何分析 1. 梳理公司的用户增长模式 尽管不同业务形态、以及不同发展阶段的公司&#xff0c;其用户增长模式各有差异&#xff0c;但都可以从拉新策略和促活策略进行分解。 常见的拉新策略有&#xff1a; 流量采购。比如通过厂商预装、…

有赞一面:亿级用户DAU日活统计,有几种方案?

说在前面 在40岁老架构师 尼恩的读者社区(50)中&#xff0c;最近有小伙伴拿到了一线互联网企业如极兔、有赞、希音、百度、网易、滴滴的面试资格&#xff0c;遇到一几个很重要的面试题&#xff1a; (1) 亿级用户场景&#xff0c;如何高性能统计日活&#xff1f; (2) 如何实现亿…

数据分析——用户粘性指标 DAU/MAU

&#xff08; 一 &#xff09;定义 DAU&#xff0c;即&#xff1a;Daily Active User&#xff0c;指日活跃用户数&#xff1b; MAU&#xff0c;即&#xff1a;Monthly Active User&#xff0c;指月活跃用户数。 日活/月活就是体现用户粘性最频繁使用的指标。日活跃用户占月活…

数据分析体系 - 用户粘性(DAU/MAU 和 月人均活跃天数)

对于常见的App&#xff0c;用户粘性的取值范围就是3%~100%&#xff0c; 不同领域的App也会有不同的基准值&#xff0c; 例如移动游戏会以20%为基线&#xff0c; 而工具类App会以40%为基线。 用户粘性的两个计算指标&#xff1a; 1、DAU/MAU 2、月平均活跃人数 这里其实…

数据分析——DAU下降/上升原因分析

本文是对“DAU变动原因”问题进行的思考整理&#xff0c;仅作记录&#xff0c;欢迎讨论。 &#xff08; 一 &#xff09;思维框架 图版&#xff1a; 文版&#xff1a; 内部原因&#xff1a; 1. 数据验证 如果DAU上升或者下降&#xff0c;且非日常波动&#xff0c;需先确…

数据分析 — 用户粘性的两个计算指标(DAU/MAU和月人均活跃天数)

很多运营都了解DAU&#xff08;日活跃用户数&#xff09;和MAU&#xff08;月活跃用户数&#xff09;的重要性&#xff0c;但在某些情况下这两个数值本身并不能反映出太多问题&#xff0c;这个时候就要引用到【DAU/MAU】的概念&#xff0c;即【日活/月活】。 用户粘性的两个计…

峰值21WQps、亿级DAU,小游戏《羊了个羊》是怎么架构的?

小游戏《羊了个羊》 短短的7天内&#xff0c;DAU突破了1亿、吞吐量峰值21WQps。 《羊了个羊》运营后台数据显示&#xff0c;在短短的7天内&#xff0c;这款小游戏的DAU就突破了1亿。 要知道&#xff0c;除了王者荣耀、原神等屈指可数的现象级手游之外&#xff0c;1亿DAU是这个…

dau、mau、pcu、dnu、wau、acu、uv分别是什么意思?

dau、mau、pcu、dnu、wau、acu、uv的意思是什么?怎么分析? DAU(Daily Active User)日活跃用户数量。常用于反映网站、互联网应用或网络游戏的运营情况。 MAU(monthly active users)月活跃用户人数。是在线游戏的一个用户数量统计名词&#xff0c;数量越大意味着玩这款游戏的人…

数据化运营04 DAU、MAU、UV:谁是最有参考价值的活跃指标?

活跃类指标是重要的用户质量指标&#xff0c;代表了产品上真正的用户&#xff0c;代表了具备营销价值的用户群&#xff0c;是几乎所有产品运营的重点。 在这一讲中&#xff0c;我会主要向你介绍 UV、DAU、MAU&#xff0c;以及每日使用时长和每日打开频次等指标。 你是否思考过…

如何理解、分析DNU/DAU?(案例:DNU、DAU面积图)

前言&#xff1a;本文内容以游戏产品为基础进行讲解&#xff0c;内容为以下4部分&#xff1a; 1. 如何理解DNU和DAU 2. 如何使用python绘制DNU和DAU的面积图 3. 如何分析DNU/DAU 4. 如何计算当日DAU有多少由往日N日DAU贡献的&#xff0c;并简单计算用户生命周期&#xff1f; DN…

DAU是啥,数据指标是啥?必知必会的数据分析常识

在刚迈入数据的大门时&#xff0c;我经常对一些数据指标或者数据本身的概念很模糊&#xff0c;尤其是当跟运营、数据分析师扯需求的时候&#xff0c;会被这些密密麻麻的指标给弄糊涂。为了更好的在行业里面摸打滚爬&#xff0c;花了很多时间阅读一些指标相关的文章、书籍&#…

dau、mau、pcu、dnu、wau、acu、uv的意思是什么?怎么分析?

dau、mau、pcu、dnu、wau、acu、uv的意思是什么?怎么分析? DAU(Daily Active User)日活跃用户数量。常用于反映网站、互联网应用或网络游戏的运营情况。 MAU(monthly active users)月活跃用户人数。是在线游戏的一个用户数量统计名词&#xff0c;数量越大意味着玩这款游戏的人…

dau、mau、pcu、dnu、wau、acu、uv的意思是什么?

dau、mau、pcu、dnu、wau、acu、uv的意思是什么?怎么分析? 名词分析 DAU(Daily Active User)日活跃用户数量。常用于反映网站、互联网应用或网络游戏的运营情况。 MAU(monthly active users)月活跃用户人数。是在线游戏的一个用户数量统计名词&#xff0c;数量越大意味着玩…

Spring AOP原理和用法

AOP也就是面向切面编程&#xff0c;首先AOP不等于Spring aop&#xff0c;前者是编程所要实现的目标&#xff0c;后者仅仅是AOP的实现方式之一&#xff0c;作为一种动态注入的实现方式&#xff0c;还有一些别的例如AspectJ&#xff08;静态注入&#xff09;等 一. Spring aop使…

一篇文章详细解读Spring的AOP原理过程(Spring面向切面详解)

目录 概述 使用登录例子说明 AOP AOP&#xff08;底层原理&#xff0c;了解&#xff09; 有两种情况动态代理 第一种 有接口情况&#xff0c;使用 JDK动态代理 第二种 没有接口情况&#xff0c;使用 CGLIB 动态代理 回顾JDK静态代理 ①明星唱歌&#xff0c;经纪人替他签…

IOC/DI、AOP相关原理

文章目录 IOC/DI为什么IOC就降低了耦合性 AOP原理使用AOP的名词们概念性名词JoinPoint/TargetIntroductionProxyWeaving 在代码中有对应注解的名词AspectPointcut&#xff08;在哪儿切&#xff09;Advice&#xff08;什么时候切&#xff09; 示例代码 IOC/DI IOC全称是Inversio…

Spring——AOP原理及流程详解

AOP原理及流程详解 一、AOP结构介绍Pointcut通知原理连接点拦截器 二、Bean介入点EnableAspectJAutoProxyAspectJAutoProxyRegistrarAnnotationAwareAspectJAutoProxyCreatorAbstractAutoProxyCreator实例前执行初始化后执行循环依赖会调用 总结 三、处理切面获取所有切面其下通…