代码实现自定义类加载器

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

手把手叫你写类加载器。

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


一. 回顾类加载器的原理


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的原因了。

相关文章
|
网络协议
HP Proliant DL360 G9使用业务网卡登录ILO管理(下)
HP Proliant DL360 G9使用业务网卡登录ILO管理
HP Proliant DL360 G9使用业务网卡登录ILO管理(下)
|
存储 编解码 安全
CSDN实训 - Java模拟二次验证码(动态令牌)
CSDN实训 - Java模拟二次验证码(动态令牌)
CSDN实训 - Java模拟二次验证码(动态令牌)
|
JavaScript 前端开发
JS实现分页功能(单选按钮、全选按钮、跳转页面)
JS实现分页功能(单选按钮、全选按钮、跳转页面)
290 0
|
Java Shell API
Ansible部署Spring Boot项目
也许Ansible能给你的便捷部署带来一些灵感。 image.png 有时,我们的基础设施并不是那么的完善,也许不想搭建Gitlab,Jenkins,就是想写一个项目,然后部署到服务器,需要简单,快速,那么Ansible基本就足够了。
|
6天前
|
人工智能 JSON 监控
Claude Code 源码泄露:一份价值亿元的 AI 工程公开课
我以为顶级 AI 产品的护城河是模型。读完这 51.2 万行泄露的源码,我发现自己错了。
4310 17
|
16天前
|
人工智能 JSON 机器人
让龙虾成为你的“公众号分身” | 阿里云服务器玩Openclaw
本文带你零成本玩转OpenClaw:学生认证白嫖6个月阿里云服务器,手把手配置飞书机器人、接入免费/高性价比AI模型(NVIDIA/通义),并打造微信公众号“全自动分身”——实时抓热榜、AI选题拆解、一键发布草稿,5分钟完成热点→文章全流程!
14940 138
让龙虾成为你的“公众号分身” | 阿里云服务器玩Openclaw
|
5天前
|
人工智能 数据可视化 安全
王炸组合!阿里云 OpenClaw X 飞书 CLI,开启 Agent 基建狂潮!(附带免费使用6个月服务器)
本文详解如何用阿里云Lighthouse一键部署OpenClaw,结合飞书CLI等工具,让AI真正“动手”——自动群发、生成科研日报、整理知识库。核心理念:未来软件应为AI而生,CLI即AI的“手脚”,实现高效、安全、可控的智能自动化。
3097 8
王炸组合!阿里云 OpenClaw X 飞书 CLI,开启 Agent 基建狂潮!(附带免费使用6个月服务器)
|
7天前
|
人工智能 自然语言处理 数据挖掘
零基础30分钟搞定 Claude Code,这一步90%的人直接跳过了
本文直击Claude Code使用痛点,提供零基础30分钟上手指南:强调必须配置“工作上下文”(about-me.md+anti-ai-style.md)、采用Cowork/Code模式、建立标准文件结构、用提问式提示词驱动AI理解→规划→执行。附可复制模板与真实项目启动法,助你将Claude从聊天工具升级为高效执行系统。