本章内容根据《深入理解Java虚拟机》第7章部分内容整理
1.什么是类加载器?
在类加载阶段,有一步是“通过类的全限定名来获取描述此类的二进制字节流”,而所谓的类加载器就是实现这个功能的一个代码模块,这个动作是在Java虚拟机外部实现的,这样做可以让应用程序自己决定如何去获取所需要的类。
类加载器的作用:首先类加载器可以实现最本质的功能即类的加载动作。同时,它还能够结合java类本身来确定该类在Java虚拟机中的唯一性。用通俗的话来说就是:比较两个类是否相等,只有这两个类是由同一个类加载器加载才有意义。否则,即使这两个类是来源于同一个Class文件,只要加载它们的类加载器不同,那么这两个类必定不相等。
2.双亲委派模型
从虚拟机的角度来说,只存在两种不同的类加载器:一种是启动类加载器(Bootstrap ClassLoader),该类加载器使用C++语言实现,属于虚拟机自身的一部分。另外一种就是所有其它的类加载器,这些类加载器是由Java语言实现,独立于JVM外部,并且全部继承自抽象类java.lang.ClassLoader。
从Java开发人员的角度来看,大部分Java程序一般会使用到以下三种系统提供的类加载器:
1)启动类加载器(Bootstrap ClassLoader):负责加载JAVA_HOME\lib目录中并且能被虚拟机识别的类库到JVM内存中,如果名称不符合的类库即使放在lib目录中也不会被加载。该类加载器无法被Java程序直接引用。
2)扩展类加载器(Extension ClassLoader):按《深入理解java虚拟机》这本书上所说,该加载器主要是负责加载JAVA_HOME\lib\ext目录中的类库,但是貌似在JDK的安装目录下,没看到该指定的目录。该加载器可以被开发者直接使用。
3)应用程序类加载器(Application ClassLoader):该类加载器也称为系统类加载器,它负责加载用户类路径(Classpath)上所指定的类库,开发者可以直接使用该类加载器,如果应用程序中没有自定义过自己的类加载器,一般情况下这个就是程序中默认的类加载器。
我们的应用程序都是由这三类加载器互相配合进行加载的,我们也可以加入自己定义的类加载器。这些类加载器之间的关系如下图所示:
如上图所示的类加载器之间的这种层次关系,就称为类加载器的双亲委派模型(Parent Delegation Model)。该模型要求除了顶层的启动类加载器外,其余的类加载器都应当有自己的父类加载器。子类加载器和父类加载器不是以继承(Inheritance)的关系来实现,而是通过组合(Composition)关系来复用父加载器的代码。
双亲委派模型的工作过程为:如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次的加载器都是如此,因此所有的类加载请求都会传给顶层的启动类加载器,只有当父加载器反馈自己无法完成该加载请求(该加载器的搜索范围中没有找到对应的类)时,子加载器才会尝试自己去加载。
使用这种模型来组织类加载器之间的关系的好处是Java类随着它的类加载器一起具备了一种带有优先级的层次关系。例如java.lang.Object类,无论哪个类加载器去加载该类,最终都是由启动类加载器进行加载,因此Object类在程序的各种类加载器环境中都是同一个类。否则的话,如果不使用该模型的话,如果用户自定义一个java.lang.Object类且存放在classpath中,那么系统中将会出现多个Object类,应用程序也会变得很混乱。如果我们自定义一个rt.jar中已有类的同名Java类,会发现JVM可以正常编译,但该类永远无法被加载运行。
在rt.jar包中的java.lang.ClassLoader类中,我们可以查看类加载实现过程的代码,具体源码如下:
- protected synchronized Class<?> loadClass(String name, boolean resolve)
- throws ClassNotFoundException
- {
- // First, check if the class has already been loaded
- Class c = findLoadedClass(name);
- if (c == null) {
- 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.
- c = findClass(name);
- }
- }
- if (resolve) {
- resolveClass(c);
- }
- return c;
- }
protected synchronized Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException { // First, check if the class has already been loaded Class c = findLoadedClass(name); if (c == null) { 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. c = findClass(name); } } if (resolve) { resolveClass(c); } return c; }
通过上面代码可以看出,双亲委派模型是通过loadClass()方法来实现的,根据代码以及代码中的注释可以很清楚地了解整个过程其实非常简单:先检查是否已经被加载过,如果没有则调用父加载器的loadClass()方法,如果父加载器为空则默认使用启动类加载器作为父加载器。如果父类加载器加载失败,则先抛出ClassNotFoundException,然后再调用自己的findClass()方法进行加载。
相关推荐
5. **类加载器**:系统类加载器、扩展类加载器和应用程序类加载器之间的双亲委派模型,以及自定义类加载器的实现。 6. **JVM调优**:通过调整JVM参数,如-Xms、-Xmx设置堆大小,-XX:NewRatio设定新生代与老年代比例...
Java类加载器采用双亲委派模型,其流程如下: 1. 如果一个类加载器收到了类加载请求,它首先不会自己去尝试加载这个类,而是把请求委托给父类加载器完成。 2. 父类加载器如果无法完成加载,则会返回给子类加载器,...
1. 类加载机制:JVM按照类加载器、双亲委派模型、验证、准备、解析、初始化等步骤加载类。了解这一过程有助于理解类的生命周期和防止类冲突。 2. 字节码执行引擎:JVM通过解释器和即时编译器(JIT)来执行字节码,...
- 双亲委派模型:类加载器在加载类时,会将任务委托给父类加载器,直到Bootstrap ClassLoader。 3. **内存区域**: - 程序计数器:记录当前线程执行的字节码指令地址。 - Java虚拟机栈:每个方法对应一个栈帧,...
9. **类加载器体系**:包括bootstrap loader、extension loader、system loader和自定义loader,它们遵循双亲委派模型,保证类加载的唯一性。 10. **JVM优化实践**:包括减少Full GC次数、调整堆大小、优化字符串池...
类加载机制遵循双亲委派模型,即当一个类加载器收到加载类的请求时,它首先会委托父类加载器尝试加载,只有当父类加载器无法加载时,子类加载器才会尝试加载。这样可以避免类的重复加载,保证核心类库的唯一性。 4...
Java的类加载器体系采用的是双亲委派模型,这意味着当一个类加载器接收到加载类的请求时,它会首先将任务委托给父类加载器,只有当父类加载器无法找到该类时,子类加载器才会尝试自己加载。 其次,自定义类加载器是...
- 双亲委派模型:类加载器会先委托父类加载器尝试加载,若失败则由子类加载器加载。 4. **垃圾收集(GC)** - **GC目标**:回收无用的对象,避免内存泄露。 - **GC类型**: Minor GC(新生代GC)、Major GC(老...
4. **类加载机制**:理解类加载的双亲委派模型,以及如何控制类加载器的行为,可以帮助我们避免类冲突和提高应用的灵活性。 5. **线程与并行性能**:监控和调整线程池大小,优化并发执行效率,例如使用`-XX:...
JVM的类加载器按照双亲委派模型工作,确保类的唯一性。加载过程包括加载、验证、准备、解析和初始化五个阶段,每个阶段都有特定的任务,例如验证字节码的安全性和正确性。 三、内存区域 JVM内存分为堆内存(Heap)...
而类加载优化涉及到类的双亲委派模型和类加载器的自定义,有助于模块化和性能提升。 最后,思维导图作为一种有效的学习工具,可以帮助我们将复杂的JVM和性能优化知识梳理清晰,形成直观的知识框架。通过xmind这样的...
本思维导图及学习笔记将深入探讨JVM的工作原理、内存模型、垃圾收集机制以及性能优化等方面,帮助你全面理解这个至关重要的技术。 一、JVM概述 Java虚拟机是Java平台的一部分,它负责解析字节码并执行Java程序。JVM...
双亲委派模型是类加载的核心机制,保证了类加载的安全性。 7. **JIT编译**:JVM提供了即时编译(Just-In-Time)技术,将经常执行的热点代码编译为本地机器码,提高运行效率。分层编译(Tiered Compilation)是现代...
1. 类加载机制:理解双亲委派模型,以及自定义类加载器的创建。 2. 内存管理:探讨堆内存,栈内存,方法区,以及垃圾回收机制。 十、设计模式 1. 创建型模式:单例、工厂、抽象工厂、建造者、原型等模式。 2. 结构...
5. **类加载机制**:了解双亲委派模型,理解类装载、验证、准备、解析和初始化的过程。 6. **性能监控和诊断工具**:如JConsole、VisualVM、JProfiler等,它们可以帮助开发者监控JVM状态,定位性能瓶颈。 7. **JVM...
5. **类加载机制**:JVM的双亲委派模型保证了类加载的有序性和安全性。理解类加载器的层次结构,以及类加载和卸载的规则,对于优化类的加载性能有帮助。 6. **JVM调优**:通过JMX(Java Management Extensions)和...
双亲委派模型确保了类加载的安全性和有序性。 3. **运行时数据区** JVM在运行过程中会划分多个区域,如堆内存、方法区、栈、程序计数器和本地方法栈。每个区域都有其特定功能,例如,堆内存存储对象实例,栈存储...
- 深入理解类加载过程,包括双亲委派模型和自定义类加载器。 - 了解方法区的结构,尤其是Java 8之后的元空间(Metaspace)。 - 理解栈帧的工作原理,包括局部变量表、操作数栈和动态链接等。 总之,"JVM成神之路"是...
- **双亲委派机制**: 如果一个类加载器收到了类加载请求,它首先不会自己去尝试加载这个类,而是把这个请求委托给父类加载器去完成,每一个层次的类加载器都是如此,因此所有的加载请求最终都应该传送到顶层的启动类...
4. **类加载机制**:理解类加载器如何加载类,双亲委派模型以及自定义类加载器的应用。 在"常用设计模式.docx"中,你将学习到一些常见的设计模式,这些模式是解决软件设计中常见问题的最佳实践: 1. **单例模式**...