Java 类加载器(ClassLoader)

Java 中的类加载器是用于加载 .class 文件到 JVM 中的组件,它的核心作用是将字节码(.class 文件)加载到内存,并且使它能够被 JVM 执行。类加载器决定了类的加载顺序和加载方式,是 Java 应用程序在运行时执行的重要部分。

1. 类加载器的基本概念

  • 类加载器:Java 中的类加载器用于将 .class 文件加载到 JVM 中,并将其转化为一个 Class 对象,之后该 Class 对象可以用于反射等操作。
  • 每个类加载器都具备加载类的能力,但它们的加载范围和加载顺序可能不同。

2. 类加载器的类型

2.1 根类加载器(Bootstrap ClassLoader)

  • 作用:是最顶层的类加载器,负责加载 Java 核心库中的类,如 java.lang.*java.util.*
  • 实现:Bootstrap ClassLoader 是由 C++ 实现的,通常由操作系统提供,加载 jre/lib 目录下的类库。
  • 加载路径:由环境变量 sun.boot.class.path 指定。

2.2 扩展类加载器(Extension ClassLoader)

  • 作用:负责加载 JDK 中的扩展类库,主要加载 jre/lib/ext 目录下的类库。
  • 实现:它是由 Java 实现的,继承自 ClassLoader
  • 加载路径:由 java.ext.dirs 指定。

2.3 系统类加载器(System ClassLoader)

  • 作用:也叫应用类加载器,负责加载应用程序的类路径(classpath)下的类。
  • 实现:由 Java 实现,通常是用户应用程序使用的加载器,加载 classpath 中的类。
  • 加载路径:由环境变量 java.class.path 指定。

2.4 自定义类加载器(Custom ClassLoader)

  • 作用:开发人员可以自定义类加载器,继承 ClassLoader 类,重写 findClass() 方法来加载类。
  • 应用场景
    • 加载本地文件、数据库、网络等外部资源中的类。
    • 用于 Web 服务器(如 Tomcat)中动态加载 JSP 文件或动态加载插件等。

3. 类加载的过程

3.1 加载(Load)

  • 类加载器根据类名查找 .class 文件,并读取它的字节流。
  • 这个过程会依赖于类加载器的父子关系,父加载器优先加载。

3.2 链接(Linking)

链接分为三个阶段:

  1. 验证(Verify):检查字节码是否符合 JVM 的要求,防止非法代码进入。
  2. 准备(Prepare):为类的静态变量分配内存并赋默认值。
  3. 解析(Resolve):将常量池中的符号引用替换为直接引用。

3.3 初始化(Initialize)

  • 在类的初始化阶段,JVM 执行类的静态代码块(static 块),初始化静态变量等。

4. 类加载器的双亲委派机制(Parent Delegation Model)

  • 双亲委派模型是 Java 类加载器的一个重要特点,它确保了类加载的安全性和稳定性。
  • 原理
    • 每个类加载器都有一个父类加载器。当一个类加载请求发生时,子加载器会先将请求传递给父加载器,父加载器先尝试加载该类。如果父加载器加载失败,子加载器再尝试加载。
    • 这确保了 Java 核心类库(如 java.lang.*)始终由根类加载器加载,从而避免了重复加载和潜在的冲突。

5. 自定义类加载器的示例

通过继承 ClassLoader 类,我们可以创建自己的类加载器,下面是一个简单的自定义类加载器示例:

public class MyClassLoader extends ClassLoader {
@Override
public Class<?> findClass(String name) throws ClassNotFoundException {
byte[] classData = loadClassData(name); // 从某个源(如文件、数据库)获取字节数据
return defineClass(name, classData, 0, classData.length);
} private byte[] loadClassData(String name) {
// 加载类的字节数据(例如,从文件或网络中读取)
return new byte[0]; // 示例返回空字节数组
}
} public class Test {
public static void main(String[] args) throws Exception {
MyClassLoader loader = new MyClassLoader();
Class<?> clazz = loader.loadClass("com.example.MyClass");
Object obj = clazz.newInstance();
}
}

解释:

  • MyClassLoader 继承自 ClassLoader,重写 findClass 方法来从某个地方加载类的字节数据。
  • loadClassData 方法是我们用来加载类字节数据的地方,可以从文件、数据库等地方加载。
  • defineClass 方法将字节数据转换为一个 Class 对象。

6. 类加载器的应用场景

  • 动态加载插件:在应用程序运行时,动态加载新的功能模块或插件,避免重新启动应用。
  • Web 服务器:如 Tomcat 会使用不同的类加载器来加载 Web 应用中的类,保证每个应用使用独立的类加载器。
  • 类版本控制:在某些情况下,自定义类加载器可以用来控制不同版本的类加载,避免类冲突。

7. 总结

  • 类加载器是 Java 中用于加载 .class 文件的组件,分为根类加载器、扩展类加载器、系统类加载器和自定义类加载器。
  • 双亲委派机制保证了类加载的顺序和安全性。
  • 自定义类加载器为我们提供了灵活性,可以从本地文件、网络等加载类,支持插件化开发。

你了解 Java 的类加载器吗?的更多相关文章

  1. java自定义类加载器

    前言 java反射,最常用的Class.forName()方法.做毕设的时候,接收到代码字符串,通过 JavaCompiler将代码字符串生成A.class文件(存放在classpath下,也就是ec ...

  2. Java虚拟机类加载器及双亲委派机制

    所谓的类加载器(Class Loader)就是加载Java类到Java虚拟机中的,前面<面试官,不要再问我"Java虚拟机类加载机制"了>中已经介绍了具体加载class ...

  3. 深入探讨java的类加载器

    类加载器是 Java 语言的一个创新,也是 Java 语言流行的重要原因之一.它使得 Java 类可以被动态加载到 Java 虚拟机中并执行.类加载器从 JDK 1.0 就出现了,最初是为了满足 Ja ...

  4. java 中类加载器

    jar 运行过程和类加载机制有关,而类加载机制又和我们自定义的类加载器有关,现在我们先来了解一下双亲委派模式. java 中类加载器分为三个: BootstrapClassLoader 负责加载 ${ ...

  5. java高新技术-类加载器

    1.类加载器及委托机制的深入分析 > 类加载器的作用:一个java文件中的出现的类,首先要把这个类的字节码加载到内存中,这个类的信息放在硬盘的classPath下的class文件中,  把cla ...

  6. 深入理解Java虚拟机 - 类加载器

    引子:       类加载器(classloader)是独立于虚拟机之外,可以独立实现的代码模块.     OSGi使用了类加载器的这一特点实现其热插拔的特性       Java同C++等语言不通, ...

  7. 分析Java的类加载器与ClassLoader(二):classpath与查找类字节码的顺序,分析ExtClassLoader与AppClassLoader的源码

    先回顾一下classpath classpath的作用: classpath的作用是指定查找类的路径:当使用java命令执行一个类(类中的main方法)时,会从classpath中进行查找这个类. 指 ...

  8. java面向对象--类加载器及Class对象

    类加载器 jvm 和 类的关系 当调用 java命令运行一个java程序时,会启动一个java虚拟机进程.同一个jvm的所有线程.所有变量都处于同一个进程里,都使用该jvm进程的内存区. jvm进程终 ...

  9. Java的类加载器

    一.类加载器的概念 类加载器(class loader)用来加载 Java 类到 Java 虚拟机中.一般来说,Java 虚拟机使用 Java 类的方式如下:Java 源程序(.java 文件)在经过 ...

  10. Java的类加载器种类(双亲委派)

    Java类加载器采用双亲委派模型: 1.启动类加载器:这个类加载器负责放在<JAVA_HOME>\lib目录中的,或者被-Xbootclasspath参数所指定的路径中的,并且是虚拟机识别 ...

随机推荐

  1. String类的使用1

    /*String:字符串,使用一对""引起来表示.1.String声明为final的,不可被继承2.String实现了Serializable接口:表示字符串是支持序列化的. 实现 ...

  2. 浅说TCP状态机制

    本文分享自天翼云开发者社区<浅说TCP状态机制>,作者:云云生息 TCP(Transmission Control Protocol)是一种面向连接的.可靠的传输协议,常用于互联网中应用层 ...

  3. 五分钟搭建属于你的AI助手:Ollama+DeepSeek+AnythingLLM深度整合教程

    作者简介 微信公众号:密码应用技术实战 博客园首页:https://www.cnblogs.com/informatics/ GitHub地址:https://github.com/warm3snow ...

  4. hibernate的锁机制

    概述 hibernate 可以通过加锁解决并发问题. hibernate 的锁分为两种:乐观锁和悲观锁. 乐观锁(Optimistic lock):每次访问数据时,都会乐观的认为其它事务此时肯定不会同 ...

  5. RabbitMQ(二)——模式类型

    RabbitMQ系列 RabbitMQ(一)--简介 RabbitMQ(二)--模式类型 RabbitMQ(三)--简单模式 RabbitMQ(四)--工作队列模式 RabbitMQ(五)--发布订阅 ...

  6. mysql数据库表如何设计

    单表数据量 所有表都需要添加注释,数据量建议控制在3000万以内 不保存大字段数据 不在数据库中存储图片.文件等大数据 表使用规范 拆分大字段和访问频率低的字段,分离冷热数据 单表字段数控制在 20 ...

  7. Flume - [05] Hbase sink

    一.概述   此接收器将数据写入Hbase.Hbase配置是从类路径中遇到的第一个Hbase-site.xml获取的.由配置指定的实现 HbaseEventSerializer 的类用于将事件转换为 ...

  8. Hadoop - 执行start-dfs.sh、stop-dfs.sh 报错处理

    执行 sbin/start-dfs.sh 和 sbin/stop-dfs.sh 报错,且进程仍然在 start-dfs.sh和stop-dfs.sh会去hadoop-env.sh中找JDK的值,但是设 ...

  9. vue2打包部署到nginx,解决路由history模式下页面空白问题

    项目使用的是vue2,脚手架vue-cli 4. 需求:之前项目路由使用的是hash,现在要求调整为history模式,但是整个过程非常坎坷,遇到了页面空白问题.现在就具体讲一下这个问题. 首先,直接 ...

  10. Go语言之sync包 WaitGroup的使用

    WaitGroup 是什么以及它能为我们解决什么问题? WaitGroup在go语言中,用于线程同步,单从字面意思理解,wait等待的意思,group组.团队的意思,WaitGroup就是指等待一组, ...