Java类加载的双亲委派机制

简介: Java类加载的双亲委派机制

一、类加载


在JVM有 两种类型的类加载器,一种是由c++编写,一种是由java编写的,其中启动类加载器(Bootstrap  Class Loader)是由c++编写,其余的都是有java编写,由java编写的类加载器都继承自类java.lang.ClassLoader,为什么小编要这么区分呢,这是因为c++编写的,开发者无法直接获取到, 所以不允许直接通过引用进行操作。下面我们分别介绍几个类加载器:


 1、启动类(Bootstrap)加载器:是由本地代码实现的类加载器,他负责将<Java_Runtime_Home>/lib下面的类库加载到内存中,例如rt.jar。由于引导类加载器涉及到虚拟机本地实现细节,开发者无法直接获取到启动类加载器的引用,所以不允许直接通过引用进行操作。其实,JVM将c++处理类加载的一套逻辑定义为启动类加载器,它是没有实体的。


2、标准扩展(Extension)类加载:是由Sun的ExtClassLoader(sun.misc.Launcher$ExtClassLoader)实现的。 它负责将<Java_Runtime_Home>/lib/ext或者由系统变量java.ext.dir指定位置中的类库加载到内存中。开发者可以直接使用标准扩展类加载器。


3、系统(System) 类加载器:是由Sun的AppClassLoader(sun.misc.Launcher$AppClassLoader)实现的。它负责系统类路径(CLASSPATH)中指定地方类库加载到内存中。 开发者可以直接使用系统类加载器。


各种类加载器存在逻辑上的父子关系,但是并不是真正的父子关系,因为他们之间没有从属关系。


20210424175847256.png



除了上面列举的 三种类加载器,还有一种比较特殊的类加载—— 线程上下文加载器


二、双亲委派机制描述


如果一个类加载器接收到加载某个类的请求,首先将加载任务委托给父类加载器,一次递归,如果父类加载器可以完成加载任务,就成功返回;只有再父类无法完成此加载任务时,才自己去加载。下面我们用一张图类描述整个过程:



0fae56a252bf78c26184bd57af0bf379.png


三、为什么需要双亲委派机制——防止内存中出现多份同样的字节码?


在回到这个问题之前我们需要知道 类加载加载的类在jvm中如何存储?


74e69ee19ed4a339adc07cbe62085e0f.png



从上面的图中我们可以看出,不同的类加载加载类在都在不同的区域,也就说如果我们同一个类使用不同的 类加载器加载,那么在jvm 中就会存在多份同样的字节码, 这种做法无疑是一种非常严重的资源浪费。这个时候如果我们使用 双亲委派机制,在加载某个的类的时候,就会递归向上查找,也就是首选使用Bootstra类加载器尝试加载,如果找不到再向下。这样能保证同一个类使用同一个类加载器加载。


四、打破双亲委派


因为在某些情况下父类加载器需要委托子类加载器去加载class文件。受到加载范围的限制,父类加载器无法加载到需要的文件,以Driver接口为例,由于Driver接口定义在jdk当中的,而其实现由各个数据库的服务商来提供,比如mysql的就写了MySQL Connector,那么问题就来了,DriverManager(也由jdk提供)要加载各个实现了Driver接口的实现类,然后进行管理,但是DriverManager由启动类加载器加载,只能记载JAVA_HOME的lib下文件,而其实现是由服务商提供的,由系统类加载器加载,这个时候就需要启动类加载器来委托子类来加载Driver实现,从而破坏了双亲委派。


既然我们要打破双亲委派,哪肯定是因为这种机制存在一定的弊端,从上面的分析我们都知道这种机制只能向上委派,所以我们要打破这种机制无非两种思路, 向下委派或者不委派。


SPI机制是一种服务发现机制。它通过在ClassPath路径下的META-INF/services文件夹查找文件,自动加载文件里所定义的类。这一机制为很多框架扩展提供了可能,比如在Dubbo、JDBC中都使用到了SPI机制。这就是打破双亲委派的一种手段,直接通过约定的路径直接加载类。


同时我们还可以通过线程上下文加载器实现向下委派,可以通过线程上下文加载器直接制定类加载

//获取
Thread.currentThread().getContextClassLoader()
// 设置
Thread.currentThread().setContextClassLoader(new Classloader_4());

五、沙箱安全——防止恶意代码污染java源代码


比如我定义了一个类名为String所在包为java.lang,因为这个类本来是属于jdk的,如果没有沙箱安全机制的话,这个类将会污染到我所有的String,但是由于沙箱安全机制,所以就委托顶层的bootstrap加载器查找这个类,如果没有的话就委托extsion,extsion没有就到aapclassloader,但是由于String就是jdk的源代码,所以在bootstrap那里就加载到了,先找到先使用,所以就使用bootstrap里面的String,后面的一概不能使用,这就保证了不被恶意代码污染。


ff6094ee0d546ec05ee9342685658060.png


目录
相关文章
|
24天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
64 2
|
29天前
|
Java 编译器
探索Java中的异常处理机制
【10月更文挑战第35天】在Java的世界中,异常是程序运行过程中不可避免的一部分。本文将通过通俗易懂的语言和生动的比喻,带你了解Java中的异常处理机制,包括异常的类型、如何捕获和处理异常,以及如何在代码中有效地利用异常处理来提升程序的健壮性。让我们一起走进Java的异常世界,学习如何优雅地面对和解决问题吧!
|
7天前
|
Java 程序员
深入理解Java异常处理机制
Java的异常处理是编程中的一块基石,它不仅保障了代码的健壮性,还提升了程序的可读性和可维护性。本文将深入浅出地探讨Java异常处理的核心概念、分类、处理策略以及最佳实践,旨在帮助读者建立正确的异常处理观念,提升编程效率和质量。
|
8天前
|
Java 开发者 UED
深入探索Java中的异常处理机制##
本文将带你深入了解Java语言中的异常处理机制,包括异常的分类、异常的捕获与处理、自定义异常的创建以及最佳实践。通过具体实例和代码演示,帮助你更好地理解和运用Java中的异常处理,提高程序的健壮性和可维护性。 ##
27 2
|
9天前
|
Java 开发者
Java中的异常处理机制深度剖析####
本文深入探讨了Java语言中异常处理的重要性、核心机制及其在实际编程中的应用策略,旨在帮助开发者更有效地编写健壮的代码。通过实例分析,揭示了try-catch-finally结构的最佳实践,以及如何利用自定义异常提升程序的可读性和维护性。此外,还简要介绍了Java 7引入的多异常捕获特性,为读者提供了一个全面而实用的异常处理指南。 ####
26 2
|
11天前
|
Java 程序员 UED
深入理解Java中的异常处理机制
本文旨在揭示Java异常处理的奥秘,从基础概念到高级应用,逐步引导读者掌握如何优雅地管理程序中的错误。我们将探讨异常类型、捕获流程,以及如何在代码中有效利用try-catch语句。通过实例分析,我们将展示异常处理在提升代码质量方面的关键作用。
25 3
|
12天前
|
Java 数据库连接 开发者
Java中的异常处理机制:深入解析与最佳实践####
本文旨在为Java开发者提供一份关于异常处理机制的全面指南,从基础概念到高级技巧,涵盖try-catch结构、自定义异常、异常链分析以及最佳实践策略。不同于传统的摘要概述,本文将以一个实际项目案例为线索,逐步揭示如何高效地管理运行时错误,提升代码的健壮性和可维护性。通过对比常见误区与优化方案,读者将获得编写更加健壮Java应用程序的实用知识。 --- ####
|
12天前
|
运维 Java 编译器
Java 异常处理:机制、策略与最佳实践
Java异常处理是确保程序稳定运行的关键。本文介绍Java异常处理的机制,包括异常类层次结构、try-catch-finally语句的使用,并探讨常见策略及最佳实践,帮助开发者有效管理错误和异常情况。
44 4
|
11天前
|
开发框架 安全 Java
Java 反射机制:动态编程的强大利器
Java反射机制允许程序在运行时检查类、接口、字段和方法的信息,并能操作对象。它提供了一种动态编程的方式,使得代码更加灵活,能够适应未知的或变化的需求,是开发框架和库的重要工具。
32 2
|
16天前
|
Java
深入探讨Java中的中断机制:INTERRUPTED和ISINTERRUPTED方法详解
在Java多线程编程中,中断机制是协调线程行为的重要手段。了解和正确使用中断机制对于编写高效、可靠的并发程序至关重要。本文将深入探讨Java中的`Thread.interrupted()`和`Thread.isInterrupted()`方法的区别及其应用场景。
21 4