当前位置: 首页 > news >正文

Java类加载器实现机制详细笔记

1. 类加载器的基本概念
  • 类加载器(ClassLoader):在Java中,类加载器负责将Java类动态加载到JVM中。它是实现动态类加载机制的核心组件,对于开发复杂应用程序(如插件系统、模块化设计等)至关重要。
2. 类加载过程
  • 加载(Loading):从文件系统或网络读取.class文件,创建包含类数据的Class对象。
  • 链接(Linking):将类的二进制数据合并到JVM中,包括:
    • 验证(Verification):确保类文件符合JVM规范。
    • 准备(Preparation):为静态变量分配内存并初始化默认值。
    • 解析(Resolution):将符号引用替换为直接引用。
  • 初始化(Initialization):为静态变量赋予正确的初始值,并执行静态代码块。
3. 双亲委派模型
  • 模型概述:Java类加载器遵循双亲委派模型(Parent Delegation Model),确保核心类库的加载安全性,避免类冲突。
  • 工作机制:类加载器收到类加载请求时,先委托给父类加载器,父类加载器找不到时,再自行加载。
4. 类加载器层次结构
  • Bootstrap ClassLoader:最顶层的类加载器,用本地代码实现,负责加载核心Java类库(如java.lang.*)。
  • Extension ClassLoader:加载扩展目录(JAVA_HOME/lib/ext)中的类。
  • Application ClassLoader:加载系统类路径(classpath)下的类,是默认的类加载器。
5. 常见类加载器
  • Bootstrap ClassLoader:由JVM实现,加载JRE核心类库。
  • Extension ClassLoader:继承自ClassLoader类,加载扩展目录中的类。
  • Application ClassLoader:继承自ClassLoader类,加载用户类路径下的类。
6. 自定义类加载器
  • 应用场景
    • 插件系统:动态加载和卸载插件,避免类冲突。
    • 热部署:不重启应用更新代码。
    • 隔离环境:隔离不同组件或模块。
    • 从非标准源加载类:如数据库、网络、加密文件。
    • 安全考虑:加载加密的类文件并解密。
7. 自定义类加载器示例
  • 示例代码
    public class CustomClassLoader extends ClassLoader {private String classPath;public CustomClassLoader(String classPath) {super(null); // 不使用默认父类加载器this.classPath = classPath;}@Overrideprotected Class<?> findClass(String name) throws ClassNotFoundException {byte[] classData = loadClassData(name);if (classData == null) {throw new ClassNotFoundException();} else {return defineClass(name, classData, 0, classData.length);}}@Overridepublic Class<?> loadClass(String name) throws ClassNotFoundException {if (name.startsWith("java.")) {return super.loadClass(name); // 委托给Bootstrap ClassLoader加载}try {return findClass(name); // 尝试自己加载类} catch (ClassNotFoundException e) {return super.loadClass(name); // 如果失败,委托给父类加载器}}private byte[] loadClassData(String className) {String filePath = classPath + className.replace('.', '/') + ".class";try (InputStream inputStream = new FileInputStream(filePath);ByteArrayOutputStream byteStream = new ByteArrayOutputStream()) {int nextValue;while ((nextValue = inputStream.read()) != -1) {byteStream.write(nextValue);}return byteStream.toByteArray();} catch (IOException e) {e.printStackTrace();return null;}}public static void main(String[] args) {String classPath = "path_to_classes/";CustomClassLoader customClassLoader = new CustomClassLoader(classPath);try {Class<?> clazz = customClassLoader.loadClass("com.example.MyClass");Object instance = clazz.newInstance();System.out.println(instance.getClass().getName());} catch (ClassNotFoundException | InstantiationException | IllegalAccessException e) {e.printStackTrace();}}
    }
    
8. 字节码校验
  • 文件格式校验
    • 检查文件头的魔数:0xCAFEBABE。
    • 验证版本号。
    • 检查常量池。
  • 元数据校验
    • 访问标志、继承关系、字段和方法描述符。
  • 字节码校验
    • 操作数栈校验、局部变量表校验、类型检查、控制流检查。
  • 符号引用校验
    • 类引用、字段和方法引用。
  • 权限校验
    • 字段和方法访问权限。
9. 魔数的重要性
  • 魔数:Class文件的前四个字节,值为0xCAFEBABE。
  • 作用:标识文件类型,防止误处理其他类型文件,确保后续解析和校验的正确性。
http://www.lryc.cn/news/407564.html

相关文章:

  • Git之repo sync -l与repo forall -c git checkout用法区别(四十九)
  • 【公式解释】《系统论》《控制论》《信息论》的共同重构:探索核心公式与深度解析
  • 电脑格式化好还是恢复出厂设置好?
  • 使用 Windows 应用程序 SDK 构建下一代应用程序
  • 可消费的媒体类型和可生成的媒体类型
  • C++中指针与迭代器的区别
  • 若依框架 : 生成代码
  • RTMP协议解析
  • 禁忌搜索算法(Tabu Search,TS)及其Python和MATLAB实现
  • Meta发布Llama 3.1 405B模型:开源与闭源模型之争的新篇章
  • Linux网络协议深度解析:从IP到TCP/IP堆栈
  • AWS DMS MySQL为源端,如何在更改分区的时候避免报错
  • Java从基础到高级特性及应用
  • JavaScript(17)——事件监听
  • Dav_笔记11:SQL Tuning Overview-sql调优 之 4
  • vue3引入openlayers
  • 大数据管理中心设计规划方案(可编辑的43页PPT)
  • Android --- 广播
  • AR 眼镜之-蓝牙电话-实现方案
  • stl-set
  • 【Stable Diffusion】(基础篇五)—— 使用SD提升分辨率
  • 5.CSS学习(浮动)
  • Spring Cloud微服务项目统一封装数据响应体
  • java算法day20
  • web自动化测试-python+selenium+unitest
  • LeetCode题练习与总结:组合两个表--175
  • 数据结构:二叉搜索树(简单C++代码实现)
  • 深入理解Prompt工程
  • 代码随想录算法训练营day6 | 242.有效的字母异位词、349. 两个数组的交集、202. 快乐数、1.两数之和
  • vue3 vxe-table 点击行,不显示选中状态,加上设置isCurrent: true就可以设置选中行的状态。