高频面试题-请把Java的双亲委派机制说清楚!

简介: 如果面试官问你,类加载过程是哪几步?巴拉巴拉巴拉...(加载、验证、准备、解析、初始化)见这小伙子面容惊奇,脸泛红光,不由自主的就问了一下双亲委派模型说一下吧;遇见没准备充分的,瞬间懵逼。下面我们就来说一下这个一个有意思的虚拟机类加载机制。

如果面试官问你,类加载过程是哪几步?


巴拉巴拉巴拉...(加载、验证、准备、解析、初始化)


见这小伙子面容惊奇,脸泛红光,不由自主的就问了一下双亲委派模型说一下吧;遇见没准备充分的,瞬间懵逼。


下面我们就来说一下这个一个有意思的虚拟机类加载机制。


一说起双亲委派,就必然要先聊一下Java中的类加载器。


Java中的类加载器

Bootstrap ClassLoader (启动类加载器)


Bootstrap ClassLoader,启动类加载,默认加载的是jdk\lib目录下jar中诸多类;

这个路径可以使用 -Xbootclasspath参数指定。


Extension ClassLoader (扩展类加载器)


Extension ClassLoader,扩展类加载器,默认加载jdk\lib\ext\目录下jar中诸多类;

这个路径可以使用 java.ext.dirs系统变量来更改。


Application ClassLoader (应用程序类加载器)


Application ClassLoader,应用程序类加载器,负责加载开发人员所编写的诸多类。


User ClassLoader (自定义类加载器)


自定义类加载器,当存在上述类加载器解决不了的特殊情况,或存在特殊要求时,可以自行实现类加载逻辑。


关系如图所示:

image.png

双亲委派模型是什么?

说完了类加载器,下面我们就说一下什么是双亲委派模型吧。


其是在JDK1.2期间被引入的,而后陆续被推荐给开发者,到目前已经成为了最常用的类加载器实现方式了。


双亲委派整个过程分为以下几步:


  1. 假设用户刚刚摸鱼写的Test类想进行加载,这个时候首先会发送给应用程序类加载器AppCloassLoader;


  1. 然后AppClassLoader并不会直接去加载Test类,而是会委派于父类加载器完成此操作,也就是ExtClassLoader;


  1. ExtClassLoader同样也不会直接去加载Test类,而是会继续委派于父类加载器完成,也就是BootstrapClassLoader;


  1. BootstrapClassLoader这个时候已经到顶层了,没有父类加载器了,所以BootstrapClassLoader会在jdk/lib目录下去搜索是否存在,因为这里是用户自己写的Test类,是不会存在于jdk下的,所以这个时候会给子类加载器一个反馈。


  1. ExtClassLoader收到父类加载器发送的反馈,知道了父类加载器并没有找到对应的类,爸爸靠不住,就只能自己来加载了,结果显而易见,自己也不行,没办法,只能给更下面的子类加载器了。


  1. AppClassLoader收到父类加载器的反馈,顿时明白,原来爸爸虽然是爸爸,但是他终究不能管儿子的私事,所以这时候,AppClassLoader就自己尝试去加载。


  1. 结果,就这样成功了,走了一大圈,兜兜转转还是自己干。


这个并没有那么复杂,我就不画图了哈,大家如果想看图,可以去网上再搜一下,我记得有个大佬画的就很形象。


为什么要使用双亲委派模型?


使用双亲委派模型,有一个很大的好处,就是避免原始类被覆盖的问题。


比如,用户编写了一个Object类,放入程序中加载。


当没有双亲委派机制时,就会出现重复的Object类,会给开发人员造成很大的困扰,本来就只需要基于JDK开发就好了,现在还得把JDK中的类全记住,避免编写重复的类。


当存在双亲委派机制时呢,整个事情就不一样了,每次加载类时,都会遵循双亲委派机制,去问父类是否可以加载,如果可以呢,那就不需要再次加载了,这样事情就变得简单了。(老子走的路,小子不能走 》.《)


如何打破双亲委派模型?


这个问题,其实就算是双亲委派模型中最深入的问题了,最起码中高级工程师面试,问到这也就下个话题了。


这里我也简单说一下吧,最常见的也是人们常说的有两种,分别是:


  1. 在自定义类加载器中,重写loadClass方法。

为什么呢?因为如果你去看ClassLoader类的源码时,你会发现,双亲委派的核心代码就是在这个方法中的;试想,你如果重写了这个方法,自然而然的就打破了双亲委派机制了。

image.png

  1. 使用线程上下文类加载器


因为双亲委派机制不能支持SPI(Service Provider Interface), 所以才会引入线程上下文类加载器,有了它,就可以打通双亲委派模型的层次结构来反向使用类加载器来完成类加载。


这个目前也是没有办法的事,涉及到SPI的目前都在使用这种方式来完成类加载,其中就包括常用到的JNDI、JDBC、JAXB等。


这里其实还有一个可以打破双亲委派模型的情况,那就是OSGI,这个解释是在《深入理解 Java虚拟机》这本书中,可能是这本书的作者本身就对OSGI比较熟悉,而且看简介还出过一本OSGI的书。


在很多应用环境中,OSGI被当做模块化热部署实现的关键,所以,这样就说明了,OSGI一定会对类加载过程做相应的一些措施。


因为这个并不常碰到,还是大家自行去查看吧,我就不在这赘述了。


目录
相关文章
|
15天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
45 2
|
20天前
|
Java 编译器
探索Java中的异常处理机制
【10月更文挑战第35天】在Java的世界中,异常是程序运行过程中不可避免的一部分。本文将通过通俗易懂的语言和生动的比喻,带你了解Java中的异常处理机制,包括异常的类型、如何捕获和处理异常,以及如何在代码中有效地利用异常处理来提升程序的健壮性。让我们一起走进Java的异常世界,学习如何优雅地面对和解决问题吧!
|
4天前
|
Java 程序员
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
28 14
|
20天前
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
|
26天前
|
存储 缓存 Oracle
Java I/O流面试之道
NIO的出现在于提高IO的速度,它相比传统的输入/输出流速度更快。NIO通过管道Channel和缓冲器Buffer来处理数据,可以把管道当成一个矿藏,缓冲器就是矿藏里的卡车。程序通过管道里的缓冲器进行数据交互,而不直接处理数据。程序要么从缓冲器获取数据,要么输入数据到缓冲器。
Java I/O流面试之道
|
1月前
|
XML 安全 Java
Java反射机制:解锁代码的无限可能
Java 反射(Reflection)是Java 的特征之一,它允许程序在运行时动态地访问和操作类的信息,包括类的属性、方法和构造函数。 反射机制能够使程序具备更大的灵活性和扩展性
41 5
Java反射机制:解锁代码的无限可能
|
22天前
|
存储 缓存 Java
大厂面试必看!Java基本数据类型和包装类的那些坑
本文介绍了Java中的基本数据类型和包装类,包括整数类型、浮点数类型、字符类型和布尔类型。详细讲解了每种类型的特性和应用场景,并探讨了包装类的引入原因、装箱与拆箱机制以及缓存机制。最后总结了面试中常见的相关考点,帮助读者更好地理解和应对面试中的问题。
44 4
|
19天前
|
Java 数据库连接 开发者
Java中的异常处理机制及其最佳实践####
在本文中,我们将探讨Java编程语言中的异常处理机制。通过深入分析try-catch语句、throws关键字以及自定义异常的创建与使用,我们旨在揭示如何有效地管理和响应程序运行中的错误和异常情况。此外,本文还将讨论一些最佳实践,以帮助开发者编写更加健壮和易于维护的代码。 ####
|
23天前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
76 4
|
25天前
|
安全 IDE Java
Java反射Reflect机制详解
Java反射(Reflection)机制是Java语言的重要特性之一,允许程序在运行时动态地获取类的信息,并对类进行操作,如创建实例、调用方法、访问字段等。反射机制极大地提高了Java程序的灵活性和动态性,但也带来了性能和安全方面的挑战。本文将详细介绍Java反射机制的基本概念、常用操作、应用场景以及其优缺点。 ## 基本概念 ### 什么是反射 反射是一种在程序运行时动态获取类的信息,并对类进行操作的机制。通过反射,程序可以在运行时获得类的字段、方法、构造函数等信息,并可以动态调用方法、创建实例和访问字段。 ### 反射的核心类 Java反射机制主要由以下几个类和接口组成,这些类
56 2