类加载器详解
创始人
2025-05-31 05:53:06
类与类加载器的关系类加载器用于实现类的加载动作,但它在java中起到的作用远远不仅于类加载阶段。对于任何一个类来说,都需要类本身和加载这个类的类加载器一同确立其在java JVM中的唯一性。通俗来讲:比较两个类是否相等,只有两个类是由同一个类加载器加载的前提下才有意义。否则,即使两个类来自同一个class文件,但是由于加载他们的类加载器不同,那这两个类就必不相等。
这里的“相等”,包括代表类的Class对象的equals()方法,isAssignableForm()方法和,isInstance()方法的返回结果。也包括instanceOf关键字对象所属关系判定等情况。如果没有注意到类加载器的影响,在某些情况下会产生迷惑人的结果。
  1. 类加载的过程

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

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

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

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

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

  1. 类加载器的分类
    主要分为两类:

  2. JVM内置的类加载器,有Bootstrap加载器、ExtClassLoader加载器和AppClassLoader加载器 三种,分别负责加载不同目录下的.class文件

  3. 用户自定义的类加载器,负责的加载目录自己决定。

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

引导类加载器负责加载\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启动类加载器 (注:不是继承关系)

扩展类加载器负责加载\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是根加载器

三者之间是没有继承关系的。

AppClassLoader和ExtClassLoader都实现了抽象类ClassLoader。

抽象类ClassLoader有一个字段parent, AppClassLoader和ExtClassLoader通过设置该字段引用,指定父加载器。(是组合关系)

AppClassLoader 的parent指向 ExtClassLoader
ExtClassLoader 的parent指向 null,(null的原因是因为Bootstrap是C++实现的,通过代码中逻辑判断来转向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;
1
2
3
4
5
6
7
8
9
10
11
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/Bootstrap
    public MyClassLoader(ClassLoader parent, String classPath) {
    super(parent);
    this.classPath = classPath;
    }

    @Override
    protected 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.Hello
    MyClassLoader 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”;
1
2.6 注:谁来准备类加载器呢?
AppClassLoader和ExtClassLoader是Launcher的静态内部类,在程序启动时JVM会创建Launcher对象,Launcher构造器会同时会创建扩展类加载器和应用类加载器。

Launcher类

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

图片来源 : https://www.bilibili.com/video/BV16T4y1P79h?p=2

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

首先使用AppClassLoader类加载器尝试加载,AppClassLoader加载器会先检查它的缓存,查看该类是否已经被加载,有则不加载,没有则向上交给ExtClassLoader加载器。
ExtClassLoader加载器同样会先检查它的缓存,查看该类是否已经被加载,有则不加载,没有则向上交给Bootstrap加载器。
Bootstrap加载器同样会先检查它的缓存,查看该类是否已经被加载。有则不加载,没有则尝试从它负责的目录中加载,
Bootstrap加载器加载失败(不在它负责的目录范围)则向下交给ExtClassLoader加载器。
ExtClassLoader加载器会从它负责的目录中尝试加载,加载失败则向下交给AppClassLoader加载器
AppClassLoader加载器从它负责的classpath尝试加载,加载完成。
双亲委派机制的好处:

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

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

打破双亲委派机制

双亲委派机制的实现其实就是在loadClass方法中实现的。
直接调用findClass方法就可以跳过双亲委派机制,这样就可以直接加载,而不用向上委托了。
1
2
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: 22913620
System.out.println(clazz2.hashCode());//out: 29768086System.out.println(clazz1.getClassLoader()); //out: 使用的类加载器 MyClassLoader@481248
System.out.println(clazz2.getClassLoader()); //out: 使用的类加载器 MyClassLoader@1947c6b

}

  1. 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;
}

}

  1. 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: 网络上的字节码文件
    

    }

二、线程上下文类加载器
1、线程上下文类加载器,打破双亲委派加载链模式(不用父类加载器先加载,可以按照自己的想法实现)
2、线程上下文类加载器,默认是app类加载器,也可以设置改变

三、tomcat

四、osgi (面向Java的动态模块系统)(通过类隔离实现逻辑上的模块化)
l OSGI的作用

在java中,OSGI是一个实现java模块化互访的平台,我们可以理解为是一个更高级的JVM。它提供了逻辑上的模块化控制。

l OSGI对模块的定义

在OSGI中,模块称之为bundle,一个bundle在物理上而言就是一个jar包。Jar包中有一个描述jar包的信息文件,位于jar内部的META-INF目录下的MANIFEST.MF文件。OSGI通过MANIFEST这个文件获取模块的定义信息,比如模块间的互访信息,模块的版本信息等。Note:对于MANIFEST.MF文件的操作,由于这个文件有很多使用约束,比如一行不能超过72个字符,所以一般都是通过IDE工具对它进行编辑
bundle是OSGi的部署(和模块)单元。
在OSGi运行时,bundle具有如下三种状态:installed,resolved,active。
在Spring中最主要的单元模块是应用程序上下文(application context),在应用程序上下文里包括了一些bean(被Spring的应用程序上下文所管理的对象)。在OSGi bundle和Spring应用程序上下文之间有很自然的紧密联系。
使用Spring Dynamic Modules(Spring DM),一个active的bundle可以包含一个Spring 应用程序上下文,它负责在bundle里实例化、配置、组装和装饰对象(bean)。这些bean即可作为OSGi服务输出而提供给其他bundle,也能透明地注入其他OSGi服务的引用。

在这里插入图片描述

osgi最明显的缺陷

bundle尽管可以为隔离的服务建立独立生命周期管理的热部署方式,以及明确的服务导出和导入依赖能力,但是其最终基于jvm,无法对bundle对应的服务实现计算资源的隔离,一个服务的故障依然会导致整个jvm crush,这使得在一个运行时的osgi上部署模块级服务只获得了模块部署和启停隔离,服务明确依赖的好处,但是没办法实现计算节点的线性扩展,在当前分布式,微服务,网络计算的趋势下,使得osgi只适合构建单一服务节点的内部应用,但是其分离的bundle的部署负担对于微服务架构来说,有点用大炮打蚊子的臭味。

osgi与tomcat类加载的区别:

tomcat中各个应用是隔离的
osgi各个模块是隔离的,但是模块之间可以通过导出、导入方式实现通信,代码共享
需要注意的是被导入的代码还是要使用原来的类加载器加载

五、sofaark
作为开源界早负盛名的动态模块系统,基于 OSGi 规范的 Equinox、Felix 等同样具备类隔离能力,然而他们更多强调的是一种编程模型,面向模块化开发,有一整套模块生命周期的管理,定义模块通信机制以及复杂的类加载模型。作为专注于解决依赖冲突的隔离框架,SOFAArk 专注于类隔离,简化了类加载模型,因此显得更加轻量。其次在 OSGi 规范中,所有的模块定义成 Bundle 形式,作为应用开发者,他需要了解 OSGi 背后的工作原理,对开发者要求比较高。在 SOFAArk 中,定义了两层模块类型,Ark Plugin 和 Ark Biz,应用开发者只需要添加隔离的 Ark Plugin 依赖,底层的类加载模型对应用开发者俩说是透明的,基本不会带来额外的学习成本。

六、

相关内容

热门资讯

漂移是什么意思(摇杆漂移是什么... 本篇文章极速百科给大家谈谈漂移是什么意思,以及摇杆漂移是什么意思对应的知识点,希望对各位有所帮助,不...
车架号后四位是什么(车架号后四... 本篇文章极速百科给大家谈谈车架号后四位是什么,以及车架号后四位是什么在哪里看对应的知识点,希望对各位...
隐形眼镜基弧是什么意思,请问,... 隐形眼镜基弧是什么意思目录隐形眼镜基弧是什么意思请问,配隐形眼镜的时候要不要关注那个基弧?隐形眼镜基...
广西省崇左市属于什么市,祟左是... 广西省崇左市属于什么市目录广西省崇左市属于什么市祟左是地级市还是县级市崇左是南宁得直辖市吗 为什么区...
内衣尺码大小分类,内衣的型号分... 内衣尺码大小分类目录内衣尺码大小分类内衣的型号分哪几种?什么abc事什么意思?34、36是尺寸嘛?内...
几个防止卫生间反味小妙招,卫生... 几个防止卫生间反味小妙招目录几个防止卫生间反味小妙招卫生间反臭怎么办?卫生间怎么样防臭几个防止卫生间...
庄子中的成语和解释,四个出自《... 庄子中的成语和解释目录庄子中的成语和解释四个出自《庄子》的成语及解释《庄子》中的成语及解释(按篇目分...
怎么切翡翠原石(收玉石的联系方... 本篇文章极速百科给大家谈谈怎么切翡翠原石,以及收玉石的联系方式对应的知识点,希望对各位有所帮助,不要...
关于燕子的古诗,描写燕子的古诗... 关于燕子的古诗目录关于燕子的古诗描写燕子的古诗描写燕子的古诗有哪些?关于燕子的古诗关于燕子的古诗 ...
好巧不巧是什么意思,好巧不巧什... 好巧不巧是什么意思目录好巧不巧是什么意思好巧不巧什么意思?“无巧不巧”究竟何解?好巧不巧是什么意思好...
关东煮里面放什么配料啊,关东煮... 关东煮里面放什么配料啊目录关东煮里面放什么配料啊关东煮的配料关东煮需要哪些调味料呀?请问,关东煮都可...
极速进化满电出发!长安深蓝SL... 本篇文章极速百科给大家谈谈极速进化满电出发!长安深蓝SL03开启预售,以及长安蓝鲸plus新车报价对...
比亚迪f3汽车报价(比亚迪f3... 今天给各位分享比亚迪f3汽车报价的知识,其中也会对比亚迪f3价格及图片易车进行解释,如果能碰巧解决你...
两台电脑怎么共享一台打印机,两... 两台电脑怎么共享一台打印机目录两台电脑怎么共享一台打印机两台电脑如何共享一台打印机?请问一个打印机怎...
16个复韵母有哪些(16个复韵... 本篇文章极速百科给大家谈谈16个复韵母有哪些,以及16个复韵母怎么读拼音视频对应的知识点,希望对各位...
樟树有什么作用,樟树有什么作用... 樟树有什么作用目录樟树有什么作用樟树有什么作用?樟树的用途有哪些?樟树有什么作用?樟树有什么作用 ...
dazl启动子的作用,启动子和... dazl启动子的作用目录dazl启动子的作用启动子和终止子是什么作用的?dazl启动子的作用启动子的...
写字楼是干什么的 极速百科网 ... 写字楼是干什么的目录写字楼是干什么的写字楼是干什么的写字楼是干什么的 写字楼的功能介绍写字楼是干什么...
骄傲的两种解释,骄傲的意思是什... 骄傲的两种解释目录骄傲的两种解释骄傲的意思是什么?骄傲的两种解释骄傲的两种解释 “骄傲”有两个...
jp是哪个国家的缩写(jp是哪... 本篇文章极速百科给大家谈谈jp是哪个国家的缩写,以及jp是哪个国家的缩写名字对应的知识点,希望对各位...
苹果手机显示不支持此配件怎么办... 不支持此配件怎么解决 苹果iphone可能不支持此配件怎么办怎么解除不支持此配件 不支持此配件怎么解...
支付宝借呗的利息是多少,蚂蚁借... 支付宝借呗的利息是多少目录支付宝借呗的利息是多少蚂蚁借呗利息是怎么计算的蚂蚁借呗的利息是多少借呗的利...
关于兰字的词语或成语越多越好.... 关于兰字的词语或成语越多越好.目录关于兰字的词语或成语越多越好.有关兰字的成语有哪些关于兰的词语或成...
宝马m5多少钱是不是很贵呢?(... 本篇文章极速百科给大家谈谈宝马m5多少钱是不是很贵呢?,以及宝马m5li多少钱对应的知识点,希望对各...
辽宁省喀左县在哪个城市,辽宁省... 辽宁省喀左县在哪个城市目录辽宁省喀左县在哪个城市辽宁省朝阳市喀左县的邮政编码辽宁省喀左县在哪里辽宁省...
关于marcjacobs香水,... 关于marcjacobs香水目录关于marcjacobs香水marcjacobs香水(探索时尚与艺术...
四级英语考试时间分配,大学英语... 四级英语考试时间分配目录四级英语考试时间分配大学英语四级考多长时间?英语四级考试时间安排?英语四级考...
dnfbuff强化有什么用,地... dnfbuff强化有什么用目录dnfbuff强化有什么用地下城buff强化栏DNF中人物的Buff有...
幼儿园孩子新年祝福语简短,适合... 幼儿园孩子新年祝福语简短目录幼儿园孩子新年祝福语简短适合幼儿园小朋友说的新年祝福语幼儿园老师给小朋友...
正断层有哪些断层组合类型,断层... 正断层有哪些断层组合类型目录正断层有哪些断层组合类型断层的组合类型简答题 断层的类型及组合形式有哪些...