代码实现自定义类加载器

简介: 代码实现自定义类加载器

手把手叫你写类加载器。

了解了类加载器的双亲委派机制, 也知道了双亲委派机制的原理,接下来就是检验我们学习是否扎实了,来自定义一个类加载器


一. 回顾类加载器的原理


1187916-20200627191708469-1781924332.png


还是这张图,类加载器的入口是c++调用java代码创建了JVM启动器,其中的一个启动器是sun.misc.Launcher启动器。这个启动器启动并加载的AppClassLoader和ExtClassLoader。然后调用launcher.getClassLoader()方法获取loader对象, loader对象本质是一个ClassLoader,然后调用了ClassLoader的loadClass("...")方法加载类。也是在loadClass("...")方法里实现了双亲委派机制。


详细原理参考文章:https://www.cnblogs.com/ITPower/p/15363400.html


二、自定义类加载器分析



对于类加载器, 我们知道他的重点是loadClass(...)方法, 里面的双亲委派机制也是在loadClass方法里面实现的. loadClass方法里面实际上去加载类的是findClass()方法. 对于我们自定义的类加载器来说需要做到两点即可

  1. 这个自定义的类加载器继承自ClassLoader
  2. 这个类加载器要重写ClassLoader类中的findClass()方法

另外我们还可以参考AppClassLoader和ExtClassLoader来写。


三、自定义类加载器实现



下面我自己定义了一个类加载器


第一步:自定义类加载器继承自ClassLoader抽象类,然后定义一个构造方法, 用来接收要加载的类名

1187916-20200629152009284-73383910.png

第二步:重写核心方法findClass(String name)

1187916-20200629152139647-1089192493.png


这里有两步操作,

第一个是: 从类路径中读取要加载类的文件内容, 自定义

第二个是: 调用构造类的方法, 调用的系统的defineClass

接下来看看自定义的loadByte是如何实现的

1187916-20200629152443259-599081814.png


这里的实现就是找到类, 并且将类的内容读取出来, 转换成二进制的字节码, 返回

最后一部分就是如何调用了.


1187916-20200629152630903-1063796716.png

用类加载器加载类, 然后实例化, 使用反射机制调用User1 的方法sout

package com.lxl.jvm;
public class User1 {
    public void sout() {
        System.out.println("进入到User1");
    }
}

这里面System.out.println(clazz.getClassLoader().getClass().getName()); 获取当前类的类加载器, 猜一猜这里打印的会是谁?

1187916-20200629152838665-522120878.png


看到了么? 是AppClassLoader, 为什么呢?

原因是我的项目里已经有一个类User1了

8780cd319003446080b4a94149d427c5_tplv-k3u1fbpfcp-zoom-1.png


我们自定义类加载器的父类是AppClassLoader. 而程序代码中的User1刚好是被AppClassLoader加载, 因为找到了,所以就不会再去我们指定的文件夹中查找了

这就是类的双亲委派机制的特点.


那么如果我们将项目中的User1类删除掉, 这是类加载器是谁呢? 当然就是我们自定义的类加载器了.


那么问题来了, 自定义类加载器的父类为什么是AppClassLoader呢?


四. 分析自定义类加载的父类为什么是appClassLoader?



我们来看一下源码

我们自定义的类加载器, 继承自ClassLoader类加载器, 那么在调用自定义类加载器的构造方法之前, 应该先加载父类ClassLoader的无参构造函数.


首先会执行ClassLoader的无参的构造方法.

1187916-20200629153826792-23924166.png

而无参的构造方法会调用自身的构造方法

1187916-20200629153703798-1151456537.png


里面有一个parent, 我们就是要看看这个parent到底是谁呢. 来看看getSystemClassLoader()方法

1187916-20200629153953381-1525329225.png

1187916-20200629154044569-889208028.png

之前我们已经研究过getClassLoader()这个方法了, 这里面定义的loadClass是谁呢?就是AppClassLoader.

1187916-20200629154345528-175003661.png


1187916-20200629154415803-2035395491.png


这就是为什么自定义class类加载器的父类是AppClassLoader的原因了。

相关文章
|
8月前
|
前端开发 Java API
类加载器“如果我定义了一个类名与Java核心类类名相同,那它还能被加载吗?”
类加载器“如果我定义了一个类名与Java核心类类名相同,那它还能被加载吗?”
|
8月前
|
应用服务中间件
自定义类加载器
自定义类加载器
36 0
|
Java
自定义类加载器实现热加载
自定义类加载器实现热加载
105 0
|
Java 数据安全/隐私保护
Java自定义类加载器的编写步骤
Java自定义类加载器的编写步骤
80 0
|
Java 程序员
ClassLoader:类的隐式加载和显式加载
ClassLoader:类的隐式加载和显式加载
251 0
|
安全 前端开发 Java
双亲委派模型与自定义类加载器
双亲委派模型与自定义类加载器
双亲委派模型与自定义类加载器
|
缓存 前端开发 Java
37. 请你详细说说类加载流程,类加载机制及自定义类加载器 中
37. 请你详细说说类加载流程,类加载机制及自定义类加载器 中
130 0
37. 请你详细说说类加载流程,类加载机制及自定义类加载器 中
|
Java 应用服务中间件 数据库
类加载器系列(三)——如何自定义类加载器
类加载器系列(三)——如何自定义类加载器
1333 0
类加载器系列(三)——如何自定义类加载器
|
Java 编译器 API
37. 请你详细说说类加载流程,类加载机制及自定义类加载器 上
37. 请你详细说说类加载流程,类加载机制及自定义类加载器 上
100 0
|
Java
37. 请你详细说说类加载流程,类加载机制及自定义类加载器 下
37. 请你详细说说类加载流程,类加载机制及自定义类加载器 下
102 0