Java ClassLoader浅析
双亲委派
提起 java 类加载器,自然绕不开其双亲委派模型
什么是双亲委派
提起双亲委派,首先想到便是那张经典的向上委派图

一般场景下,当某个类将要被加载时,由系统上下文默认的类加载器Thread.currentThread().getContextClassLoader()对该类进行加载,通常这个类加载器为AppClassLoader,AppClassLoader不会直接尝试加载这个类,而是委托给它的父类ExtClassLoader对其加载,ExtClassLoader则会直接委托父类BootstrapClassLoader加载,当BootstrapClassLoader找不到要加载的目标类时,才会逐级下放给子类,否则类加载完毕
这里需要明确2个概念:
- 父类加载器 这里虽然称为父加载器,但却跟 java 的继承没有关系,反而这里实现“父”加载器是通过组合的方式,本文后续还会说明
- 类定义加载器 真正加载该类的类加载器,而不是发起加载动作的类加载器;比如AppClassLoader发起加载String类,而真正执行加载动作的却是BootstrapClassLoader,所以定义String的类加载器为BootstrapClassLoader
双亲委派好处
java 的设计者们为什么要大费周章设计如此复杂的加载模型呢?其实主要有以下两个好处
- 避免同一个类被重复加载 - 当用户定义了相当多的类加载器时,如果没有此委托模型,将产生大量的类重复加载,占用内存(同一个类被不同的类加载器加载时,将会产生多个副本),可参见下面用例  
- 保护 jdk 核心类不被篡改 - java向来以安全著称,试想,如果没有此隔离机制,某个依赖的 jar 包中被恶意篡改、注入 - java.lang.String类后,将导致整个进程的奔溃;同样参见上图,由于双亲委派机制,java 内部类不管由自定义类加载器还是- AppClassLoader,都仅会产生一份实例,且一定由- BootstrapClassLoader加载
jdk 类加载器剖析
下面我们依次分析3个类加载器
- AppClassLoader- 路径 - sun.misc.Launcher#AppClassLoader
- 此类加载器主要负责加载用户编写的java文件,同时java默认的线程上下文类加载器亦为此类。我们翻看 - sun.misc.Launcher源码验证此观点 
- 类结构图  
- 总结 - 在一般的工程中,此类用来加载初jdk内部类外的所有java文件,如果项目工程需要自定义类加载器的话,一般也是会继承自此类来保证双亲委派模型;不过有些做三方包、二方包隔离的中间件会自定义多个类加载器继承自 - ExtClassLoader,从而保证用户编写的代码、依赖的jar包做到真正隔离
 
- ExtClassLoader- 路径 - sun.misc.Launcher#ExtClassLoader
- 类结构图  
- 解析 - 翻看源码,发现此类的parent为null  - 而加载class的方法 - ClassLoader#loadClass定义如下
  - 即某个类加载器的parent指定为 null 时,它的父类会自动变为 - BootstrapClassLoader,而- BootstrapClassLoader定义在- native方法上
- 加载哪些类? - ExtClassLoader会加载系统变量- java.ext.dirs对应的所有路径中的 jar 包,以macos为例,- System.getProperty("java.ext.dirs")内容如下:- /Users/***/Library/Java/Extensions
 /Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/Home/jre/lib/ext
 /Library/Java/Extensions
 /Network/Library/Java/Extensions
 /System/Library/Java/Extensions
 /usr/lib/java
 
 
- BootstrapClassLoader- 加载哪些类? - BootstrapClassLoader加载java的核心类库,我们可以通过- sun.misc.Launcher.getBootstrapClassPath().getURLs()来获取其具体路径,以macos为例,内容如下:- file:/Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/Home/jre/lib/resources.jar
 file:/Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/Home/jre/lib/rt.jar
 file:/Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/Home/jre/lib/sunrsasign.jar
 file:/Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/Home/jre/lib/jsse.jar
 file:/Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/Home/jre/lib/jce.jar
 file:/Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/Home/jre/lib/charsets.jar
 file:/Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/Home/jre/lib/jfr.jar
 file:/Library/Java/JavaVirtualMachines/jdk1.8.0_151.jdk/Contents/Home/jre/classes
 
 
至此,我们可以把双亲委派的模型画的更具体一些

打破双亲委派
java 双亲委派模型似乎固若金汤,足够安全;不过我们自定义的类加载器想打破这个委派机制的话,也是相当容易,甚至 jdk 自身在处理某些场景时,自己也会打破该机制,比如赫赫有名的 SPI
如何打破
如何打破双亲委派其实非常简单,我们先看 java 是如何保证双亲委派正常运行的

ClassLoader的loadClass方法保证了双亲委派,不过此方法是被protected修饰的,所以任何子类都可以重写此方法;java 建议使用者重写findClass方法,这样可以在不破坏双亲委派的机制下,达到加载自身类的目的。不过在某些场景下,我们依旧可以直接重写loadClass来破坏委派机制
SPI
讨论 java 类加载器为何要提到spi ? 是的,讨论类加载器一定绕不开 spi 的设计理念,spi 对委派概念是一次冲击,我们也可以看到 java 设计者们在双亲委派问题上做的让步
什么是spi
我们先简单回顾下 spi 的定义;spi 是 Service Provider Interface 的简写,翻译成中文就是服务提供发现接口。
要使用Java SPI,需要遵循如下约定:
- 1、当服务提供者提供了接口的一种具体实现后,在jar包的META-INF/services目录下创建一个以“接口全限定名”为命名的文件,内容为实现类的全限定名;
- 2、接口实现类所在的jar包放在主程序的classpath中;
- 3、主程序通过java.util.ServiceLoder动态装载实现模块,它通过扫描META-INF/services目录下的配置文件找到实现类的全限定名,把类加载到JVM;
- 4、SPI的实现类必须携带一个不带参数的构造方法;
java 通过 spi 机制,在解耦的同时实现了可插拔操作
带来的问题
通过上文分析,我们知道像诸如rt.java等核心类库均交由BootstrapClassLoader加载,不过核心类库返回的对象却是用户自定义的实现类,这在双亲委派机制上是说不通的,下面我们拿jdbc驱动举例说明
首先我们看下驱动启动的代码实例
Class.forName("com.mysql.jdbc.Driver");
Connection connection = DriverManager.getConnection("jdbc:mysql://localhost:3306/wolong_extend?rewriteBatchedStatements=true&useServerPrepStmts=true","root","root");
System.out.println(connection.getClass().getName());
最后的输出结果为 com.mysql.jdbc.JDBC4Connection
mysql 对应的实现类在代码之前并未加载,而在加载 java rt.jar包中的java.sql.DriverManager时却返回了一个用户定义的对象,给人的错觉便是BootstrapClassLoader加载了com.mysql.jdbc.JDBC4Connection,那真实的情况又是什么样呢?
- 注:当 java 加载类 A,而 A 又依赖了 B,java 会用加载 A 的类加载器去加载类 B
DriverManager剖析

翻看源码发现 java 为了解决 spi 问题引入了上下文类加载器的概念Thread.currentThread().getContextClassLoader(),首先获取调用类caller的类加载器,如果为空的话,便直接获取上下文类加载器
加载顺序

后记
所有实例代码 https://github.com/xijiu/share
Java ClassLoader浅析的更多相关文章
- Android(java)学习笔记106-1:深入分析Java ClassLoader原理
		1. 前言: Android中的动态加载机制能更好的优化我们的应用,同时实现动态的更新,这就便于我们管理我们的应用,通过插件化来减轻我们的内存以及CPU消耗,在不发布新版本的情况下能更新某些模块. 当 ... 
- Java Classloader机制解析(转)
		做Java开发,对于ClassLoader的机制是必须要熟悉的基础知识,本文针对Java ClassLoader的机制做一个简要的总结.因为不同的JVM的实现不同,本文所描述的内容均只限于Hotspo ... 
- [转载] 深入了解Java ClassLoader、Bytecode 、ASM、cglib
		转载自http://www.iteye.com/topic/98178 一.Java ClassLoader 1,什么是ClassLoader 与 C 或 C++ 编写的程序不同,Java 程序并 ... 
- Java Classloader机制解析
		做Java开发,对于ClassLoader的机制是必须要熟悉的基础知识,本文针对Java ClassLoader的机制做一个简要的总结.因为不同的JVM的实现不同,本文所描述的内容均只限于Hotspo ... 
- Java ClassLoader加载机制理解 实际例子
		针对 Java ClassLoader加载机制理解, 做了个如何自定制简单的ClassLoader,并成功加载指定的类. 不废话,直接上代码. package com.chq.study.cl; im ... 
- Java ClassLoader加载机制理解
		今天看到了一篇介绍Java ClassLoader加载机器的文章, 才发觉一直来自己的肤浅, 好好地给补了一课, 不得不存档! 原文地址: http://www.blogjava.net/lhulcn ... 
- java classloader原理深究
		前面已经写过一篇关于java classloader的拙文java classloader原理初探. 时隔几年,再看一遍,觉得有些地方显得太过苍白,于是再来一篇: 完成一个Java类之后,经过java ... 
- Android(java)学习笔记45:深入分析Java ClassLoader原理
		1. 前言: Android中的动态加载机制能更好的优化我们的应用,同时实现动态的更新,这就便于我们管理我们的应用,通过插件化来减轻我们的内存以及CPU消耗,在不发布新版本的情况下能更新某些模块. 当 ... 
- 转 Java Classloader机制解析
		转 Java Classloader机制解析 发表于11个月前(2014-05-09 11:36) 阅读(693) | 评论(0) 9人收藏此文章, 我要收藏 赞1 慕课网,程序员升职加薪神器,点 ... 
随机推荐
- 从TFS到git的持续集成之路
			前言 公司目前使用TFS,由于TFS不灵活不能很好的持续集成,且给测试造成很大重的负担,所以近期准备迁移到git上 目标 解决项目运转的瓶颈(版本太多,导致测试跟不上,需引入自动化测试) 过程 主线分 ... 
- opencv-python imread、imshow浏览目录下的图片文件
			☞ ░ 前往老猿Python博文目录 ░ 一.几个知识点 1.1.使用Python查找目录下的文件 具体请参考<Python正则表达式re模块和os模块实现文件搜索模式匹配>. 1.2.o ... 
- moviepy音视频剪辑:视频半自动追踪人脸打马赛克
			一.引言 在<moviepy1.03音视频剪辑:使用manual_tracking和headblur实现追踪人脸打马赛克>介绍了使用手动跟踪跟踪人脸移动轨迹和使用headblur对人脸进行 ... 
- 第十一章  Python 支撑正则表达式处理的re模块
			re模块是Python中支持正则表达式处理的模块,老猿学了之后,发现这部分内容太多,要表述清楚需要开单章才能写清楚,但老猿觉得re模块的使用对多数人来说要通过教程学习去熟练掌握很难,需要经常接触练习加 ... 
- PyQt(Python+Qt)学习随笔:QListWidget的currentRow属性
			QListWidget的currentRow属性保存当前项的位置,为整型,从0开始计数,在某些选择模式下,当前项可能也是选中项. currentRow属性可以通过方法currentRow().setC ... 
- 第15.15节 PyQt(Python+Qt)入门学习:Designer的menu菜单、toolBar工具栏和Action动作详解
			老猿Python博文目录 老猿Python博客地址 一.引言 Qt Designer中的部件栏并没有菜单.toolBar以及Action相关的部件,仅在MainWindow类型窗口提供了menu.to ... 
- 我摊牌了,大厂面试Linux就这5个问题
			说真的,这就是<我想进大厂>系列第八篇,但是Linux的问题确实很少,就这样,强行编几个没有营养的问题也没啥意义. 1.CPU负载和CPU利用率的区别是什么? 首先,我们可以通过uptim ... 
- Acwing 120. 防线
			题目地址 题目简译:给定\(n\)个等差数列,每个等差数列的起点为\(s\),终点为\(e\),差为\(d\).整个序列中至多有一个位置所占数字是奇数.判断奇数位是否存在,如果不存在输出"T ... 
- P6772 [NOI2020]美食家
			题目大意 给你一个 \(n\) 个点,\(m\) 条边的有向图,每条边有一个权值 \(w_i\) ,每个节点有一个权值 \(a_i\) . 你从节点 \(1\) 出发,每经过一个节点就可以获得该点的权 ... 
- 题解-[国家集训队]Crash的数字表格 / JZPTAB
			题解-[国家集训队]Crash的数字表格 / JZPTAB 前置知识: 莫比乌斯反演 </> [国家集训队]Crash的数字表格 / JZPTAB 单组测试数据,给定 \(n,m\) ,求 ... 
