1、类加载器

1.1 类加载的概念

  要了解双亲委派模型,首先我们需要知道java的类加载器。所谓类加载器就是通过一个类的全限定名来获取描述此类的二进制字节流,然后把这个字节流加载到虚拟机中,获取响应的java.lang.Class类的一个实例。我们把实现这个动作的代码模块称为“类加载器”。

1.2 类与类加载器

  对于任意的一个类,都需要由加载它的类加载器和这个类本身一同建立其在Java虚拟机中的唯一性,每个类加载器,都拥有一个独立的类名称空间,即:比较两个类是否“相等”,只有在这两个类是由同一个类加载器加载的前提下才有意义,否则,即使这两个类源于同一个Class文件,被同一个虚拟机加载,只要加载它们的类加载器不同,那这两个类就必定不想等。

package com.demo.test;import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;/*** @author lxc* @createTime 2023-02-09 10:20* @description*/
public class ClassLoaderTest {public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {ClassLoader myloader = new ClassLoader() {@Overridepublic Class<?> loadClass(String name) throws ClassNotFoundException {String fileName = name.substring(name.lastIndexOf(".") + 1) + ".class";InputStream is = getClass().getResourceAsStream(fileName);if (is == null) {return super.loadClass(name);}try {byte[] b = new byte[is.available()];is.read(b);return defineClass(name,b,0,b.length);} catch (IOException e) {throw new RuntimeException(e);}}};Object obj = myloader.loadClass("com.demo.test.ClassLoaderTest").newInstance();System.out.println(obj.getClass());System.out.println(obj instanceof ClassLoaderTest);Class<?> clazz = Class.forName("com.demo.test.ClassLoaderTest");Constructor<?> constructor = clazz.getConstructor();Object obj1 = constructor.newInstance();System.out.println(obj1 instanceof ClassLoaderTest);}
}

运行结果如下:

obj对象的Class:class com.demo.test.ClassLoaderTest
obj1对象的Class:class com.demo.test.ClassLoaderTest
false
true

从运行结果来看,第一句和第二句来看,两个对象都是由类class com.demo.test.ClassLoaderTest实例化出来的对象;
从第三句可以看出,这个对象与类class com.demo.test.ClassLoaderTest做所属类型检查时却返回了false,这是因为虚拟机中存在了两个ClassLoaderTest,一个是由我们自定义的类加载器加载的,
另一个是由系统应用程序类加载器加载的,虽然都来自同一个Class文件,但依然是两个独立的类,做对象所属类型检查时结果为false。

2、双亲委派模型

2.1 类加载器的分类

  从虚拟机的角度来看,存在两种不同的类加载器:一种是启动类加载器(Bootstrap ClassLoader),这个类加载器使用C++语言实现,是虚拟机自身的一部分;另一个就是所有的其他类加载器,这些类加载器都是由Java语言实现,独立于虚拟机外部,并且全都继承字抽象类java.lang.ClassLoader。
  从开发者角度来看,类加载器可以分为以下四类:启动类加载器(Bootstrap ClassLoader)、扩展类加载器(Extension ClassLoader)、应用程序类加载(Application ClassLoader)和自定义类加载器。

  • 启动类加载器(Bootstrap ClassLoader):这个类加载器是加载核心java库,负责将<JAVA_HOME>/jre/lib目录中的,或者被-Xbootclasspath参数所指定的路径中的,并且是虚拟机识别的(仅仅按照文件名识别,如rt.jar,名字不符合的类库即使放在lib目录下也不会被加载)类库加载到虚拟机内存中。开发者不能直接使用启动类加载器。
  • 扩展类加载器(Extension ClassLoader):这个类加载器是由sun.misc.Launcher$ExtClassLoader实现,它负责加载<JAVA_HOME>/jre/lib/ext目录中的,或者被java.ext.dirs系统变量所指定的路径中的所有类库,开发者可以直接使用该类加载器。
  • 应用程序类加载器(Application ClassLoader):这个类加载器是由sun.misc.Launcher$AppClassLoader实现。这个类加载器是ClassLoader中的getSystemClassLoader()方法的返回值,所以一般也称之为系统类加载器。它负责加载用户类路径(ClassPath)上所指定的类库,开发者可以直接使用这个类加载器,如果应用程序中没有定义过自己的类加载器,一般情况下这就是程序中默认的类加载器。
      通过上面的分类我们可以看到,这三种类加载器只能加载各自所负责的目录下的类库,而不能加载超过其目录范围的类库,这也就是我们常常说的双亲委派模型中的可见性原则。
      我们平时所写的应用程序都是由这三种类加载器相互配合进行加载的,如果有必要,还可以加上自己定的类加载器。

2.2 双亲委派模型中各类加载器之间的层次关系


  类加载器之间这种层次关系,我们称之为类加载的双亲委派模型。双亲委派模型中,除了顶层的启动类加载器外,其余的类加载器都应当有自己的父-类加载。这里的父子关系不是以继承关系来实现的,而都是使用组合的关系来复用父-类加载的代码。

2.3 双亲委派模型中类加载的工作过程

  如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父-类加载去完成,每一个层次的类加载器(启动类加载器除外)都是如此,因此所有的加载请求最终都应该传送到顶层的启动类加载器中,只有当父-类加载器反馈自己无法完成这个加载请求时,子-类加载器才会尝试自己去加载。
下面我们看段源码,从代码角度看一下这个工作过程:

protected Class<?> loadClass(String name, boolean resolve)throws ClassNotFoundException{synchronized (getClassLoadingLock(name)) {// First, check if the class has already been loaded//首先检查请求的类是否被加载过Class<?> c = findLoadedClass(name);if (c == null) {//如果没有被加载过,就进行加载操作long t0 = System.nanoTime();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.//在父-类加载器无法完成加载的时候,再调用本身的findClass方法来进行类加载long t1 = System.nanoTime();c = findClass(name);// this is the defining class loader; record the statssun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);sun.misc.PerfCounter.getFindClasses().increment();}}if (resolve) {resolveClass(c);}return c;}}

  双亲委派模型对于保证java的稳定运行很重要,但从上面的源码来看,实现还是比较简单的,双亲委派模型的核心代码主要都在java.lang.ClassLoader的loadClass()方法中,大体逻辑如下:
  先检查是否已经被加载过,若没有加载则调用父-类加载的loadClass()方法,若父类加载为空则默认使用启动类加载器做父-类加载器加载。如果父-类加载器加载失败,抛出ClassNotFoundException异常后,再调用自己的findClass()方法进行加载。

2.4 双亲委派模型的目的

  那么我们思考一下,java为什么采用双亲委派模型呢?从上面双亲委派模型的工作过程,我们看出,java类随着它的类加载器一起具备了带有优先级的层次关系。例如类java.lang.Integer,它存放在核心包rt.jar中,那么无论哪一个类加载器要加载这个类,最终都
是要委派给处于最顶端的启动类加载器进行加载,从而是的Integer类在程序的各中类加载器环境中都是同一个类;相反,如果没有使用双亲委派模型,而是由各个类加载器自行去加载的话,当用户自己编写了一个名为java.lang.Integer类并放到ClassPath中,那么系统将会出现多个不同的Integer类,这样就会造成java体系中最基础的行为都无法保证(连最基本的类型都不唯一),程序将变得一片混乱。你可能会说,我自定义一个类加载去加载java.lang.Integer,直接重写loadClass方法,从而破坏掉双亲委派模型不就行了。
  我们写个简单的例子试下。

public class MyClassLoader extends ClassLoader {@Overridepublic Class<?> loadClass(String name) throws ClassNotFoundException {String className = null;if (name.startsWith("java.lang")) {className = "/" + name.replace(".", "/") + ".class";} else {className = name.substring(name.lastIndexOf(".") + 1) + ".class";}InputStream is = getClass().getResourceAsStream(className);if (is == null) {return super.loadClass(name);}try {byte[] bytes = new byte[is.available()];is.read(bytes);return defineClass(name, bytes, 0, bytes.length);} catch (IOException e) {throw new RuntimeException(e);}}public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException {ClassLoader myLoader = new MyClassLoader();Object obj = myLoader.loadClass("java.lang.Integer").newInstance();System.out.println(obj);}
}

结果输出:

Exception in thread "main" java.lang.SecurityException: Prohibited package name: java.langat java.lang.ClassLoader.preDefineClass(ClassLoader.java:662)at java.lang.ClassLoader.defineClass(ClassLoader.java:761)at java.lang.ClassLoader.defineClass(ClassLoader.java:642)at com.demo.test.MyClassLoader.loadClass(MyClassLoader.java:28)at com.demo.test.MyClassLoader.main(MyClassLoader.java:36)

  从源码分析来看,主要是defineClass方法调用的preDefineClass方法异常,在preDefineClass这个方法中我们看到:

  if ((name != null) && name.startsWith("java.")) {throw new SecurityException("Prohibited package name: " +name.substring(0, name.lastIndexOf('.')));}

即如果是以java.开头的包下的类,都只能用启动类加载器来加载。

2.5 双亲委派模型的三个原则

  双亲委派模型有三个基本原则:委托性、可见性和唯一性原则。

  • 委托性原则:当子类加载器收到类加载请求时,会将加载请求向上委托给父类加载器;
  • 可见性原则:每种类加载器都有自己可加载类库的范围,超出这个范围是不可见的,即无法加载的;
  • 唯一性原则:这是双亲委派模型的核心,也是最重要的目的。

2.6 为什么要打破双亲委派模型

  我们这里主要说一下JDBC为什么要打破双亲委派模型,其他的方面我后续再分析。
  我们以mysql数据库驱动为例来说明。最早我们使用mysql数据库驱动的时候,一般是这样写代码:

    Class.forName("com.mysql.jdbc.Driver");Connection conn = DriverManager.getConnection("jdbc:mysql://host:port/dbname?useUnicode=true&characterEncoding=utf-8&useSSL=false", "username", "password");

  其中com.mysql.jdbc.Driver下的Driver.class的源码如下:

          package com.mysql.jdbc;import java.sql.DriverManager;import java.sql.SQLException;public class Driver extends NonRegisteringDriver implements java.sql.Driver {public Driver() throws SQLException {}static {try {DriverManager.registerDriver(new Driver());} catch (SQLException var1) {throw new RuntimeException("Can't register driver!");}}}

  从com.mysql.jdbc的Driver.java源码中看到,在Driver类中向DriverManager注册了对应的驱动实现类。
  而从JDBC4.0以后,开始支持使用SPI的方式来注册这个Driver,这样当我们使用不同jdbc驱动时,就不用手动修改Class.forName加载的驱动类,只需要加入相关的jar包就行了。所以上面的数据库连接代码可以简写成如下:

Connection conn = DriverManager.getConnection("jdbc:mysql://host:port/dbname?useUnicode=true&characterEncoding=utf-8&useSSL=false", "username", "password");

这就不需要Class.forName(“com.mysql.jdbc.Driver”)了。
了解SPI的同学都知道,在DriverManager中,这时候对应的驱动类大体是这么加载的:
  1.通过从META-INF/services/java.sql.Driver文件中获取具体的实现类”com.mysql.jdbc.Driver“;
  2.通过Class.forName(“com.mysql.jdbc.Driver”)将这个类加载进来。
  但是DriverManager是在java.sql中,在rt.jar包中,这个包中的类只能使用启动类加载器进行加载,那么根据类加载的机制,当被装载的类引用了另外一个类的时候,虚拟机就会使用装载第一个类的类装载器装载被引用的类。:启动类加载器还要去加载mysql驱动jar中的类(com.mysql.jdbc.Driver),这显然是不可能的,根据双亲委派模型的可见性原则,启动类加载器找不到这个mysql类库,所以无法加载。
  这个问题更加有适用性的说法应该是:JAVA核心包中的类去调用开发者实现的类的方法,这时候就会出现启动类加载器无法加载到具体实现类的问题。所以想让启动类加载器(顶层类加载器)加载可见范围之外的类库,只能破坏双亲委派模型中的委托性原则,让启动类加载器委托-子类加载器去加载,所以破坏了双亲委派模型中向父-类加载器委托的原则。com.mysql.jdbc.Driver这个类库,其实是由Application ClassLoader系统类加载器加载完成的,只不过从表面上看起来是破坏了可见行原则,实质上加载com.mysql.jdbc.Driver这个类库并没有破坏双亲委派原则。
  下面我们看DriverManager是怎么实现的。
  DriverManager加载时,会执行静态代码块,在静态代码块中,会执行loadInitialDrivers方法。而这个方法中会加载对应的驱动类。

public class DriverManager {static {loadInitialDrivers();println("JDBC DriverManager initialized");}private static void loadInitialDrivers() {String drivers;try {drivers = AccessController.doPrivileged(new PrivilegedAction<String>() {public String run() {return System.getProperty("jdbc.drivers");}});} catch (Exception ex) {drivers = null;}AccessController.doPrivileged(new PrivilegedAction<Void>() {public Void run() {// 根据配置文件加载驱动实现类,下面这个方法中说明了所使用的类加载器ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(Driver.class);Iterator<Driver> driversIterator = loadedDrivers.iterator();try{while(driversIterator.hasNext()) {driversIterator.next();}} catch(Throwable t) {// Do nothing}return null;}});println("DriverManager.initialize: jdbc.drivers = " + drivers);if (drivers == null || drivers.equals("")) {return;}String[] driversList = drivers.split(":");println("number of Drivers:" + driversList.length);for (String aDriver : driversList) {try {println("DriverManager.Initialize: loading " + aDriver);Class.forName(aDriver, true,ClassLoader.getSystemClassLoader());} catch (Exception ex) {println("DriverManager.Initialize: load failed: " + ex);}}}
}public static <S> ServiceLoader<S> load(Class<S> service) {//使用了一个线程上下文类加载器ClassLoader cl = Thread.currentThread().getContextClassLoader();return ServiceLoader.load(service, cl);}

  ExtClassLoader和AppClassLoader都是通过Launcher类来创建的,在Launcher类的构造函数中我们可以看到线程上下文类加载器默认是AppClassLoader。Launcher类中无参构造方法:

public Launcher() {ExtClassLoader var1;try {var1 = Launcher.ExtClassLoader.getExtClassLoader();} catch (IOException var10) {throw new InternalError("Could not create extension class loader", var10);}try {this.loader = Launcher.AppClassLoader.getAppClassLoader(var1);} catch (IOException var9) {throw new InternalError("Could not create application class loader", var9);}//设置当前线程的上下文类加载器就是AppClassLoaderThread.currentThread().setContextClassLoader(this.loader);String var2 = System.getProperty("java.security.manager");if (var2 != null) {SecurityManager var3 = null;if (!"".equals(var2) && !"default".equals(var2)) {try {var3 = (SecurityManager)this.loader.loadClass(var2).newInstance();} catch (IllegalAccessException var5) {} catch (InstantiationException var6) {} catch (ClassNotFoundException var7) {} catch (ClassCastException var8) {}} else {var3 = new SecurityManager();}if (var3 == null) {throw new InternalError("Could not create SecurityManager: " + var2);}System.setSecurityManager(var3);}

2.7 如何打破双亲委派模型?

  在ClassLoader中有几个核心方法,上面我们已经展示了loadClass的基本源码,下面我们再简略看一下(去掉了一些代码细节):

    package java.lang;public abstract class ClassLoader {protected Class defineClass(byte[] b); protected Class<?> findClass(String name); protected Class<?> loadClass(String name, boolean resolve) {synchronized (getClassLoadingLock(name)) {// 1. 检查类是否已经被加载过Class<?> c = findLoadedClass(name);if (c == null) {try {if (parent != null) {//2. 委托给父类加载c = parent.loadClass(name, false);} else {//3. 父类不存在的,交给启动类加载器c = findBootstrapClassOrNull(name);}} catch (ClassNotFoundException e) { }if (c == null) {//4. 父类加载器无法完成类加载请求时,调用自身的findClass方法来完成类加载c = findClass(name);}}return c;}
}
  • defineClass 方法:调用 native 方法将 字节数组解析成一个 Class 对象;
  • findClass 方法:抽象类ClassLoader中默认抛出ClassNotFoundException,需要继承类自己去实现,目的是通过文件系统或者网络查找类;
  • loadClass 方法: 首先根据类的全限定名检查该类是否已经被加载过,如果没有被加载,那么当子加载器持有父加载器的引用时,那么委托给父加载器去尝试加载,如果父类加载器无法完成加载,再交给子类加载器进行加载。loadClass方法 就是实现了双亲委派机制。
      ClassLoader 的三个重要方法,那么如果需要自定义一个类加载器的话,直接继承 ClassLoader类,一般情况只需要重写 findClass 方法即可,自己定义加载类的路径,可以从文件系统或者网络环境。但是,如果想打破双亲委派机制,那么还要重写 loadClass 方法。

java的双亲委派模型-附源码分析相关推荐

  1. java手机象棋软件下载,Java手机网络版象棋游戏附源码JAVA游戏源码下载

    运行于手机上的中国象棋游戏,鉴于JAVA J2ME技术,本游戏分服务端和客户端,导入JAR包即可在手机上运行,内有开发文档和运用说明. Java手机网络版象棋游戏附源码 (1 folders, 2 f ...

  2. 天天酷跑php源码_使用Java实现天天酷跑(附源码)

    首先,写一个需求文档: 一.项目名称:<天天酷跑>(RunDay) 二.功能介绍: 闯关类游戏,玩家登录后,选择进入游戏,通过键盘控制玩家的上下左右移动,来躲避 障碍物和吃金币,玩家躲避的 ...

  3. 百看不如一练,55个Java练手项目(附源码+视频教程),全都在这里了

    我们都知道,不管学习那门语言最终都要做出实际的东西来,而对于编程而言,这个实际的东西当然就是项目啦,不用我多说大家都知道学编程语言做项目的重要性. 于是,我熬了几个通宵,终于整理出了55个培训机构内部 ...

  4. Java项目—健身房管理系统(附源码+文档)

    今天给大家分享一个Java实战项目-健身房管理系统(附源码+数据库+文档) 需要资料源码的小伙伴可以点击下方链接和小编一起学习~ 需要源码直接私信小编哟~https://www.bilibili.co ...

  5. 手把手搭建Java金融借贷系统【附源码】(毕设)

    一.项目简介 本课程演示的是一套基于基于JavaWeb实现的金融借贷系统 或 P2P金融管理系统 或 小额贷款系统,主要针对计算机相关专业的正在做毕设的学生与需要项目实战练习的java人群. 详细介绍 ...

  6. 视频教程-手把手搭建Java金融借贷系统【附源码】(毕设)-Java

    手把手搭建Java金融借贷系统[附源码](毕设) 南京大学软件工程硕士,全栈开发工程师,全栈讲师. 曾就职于中软国际.擎天科技.华为等公司,擅长Java开发.Web前端.Python爬虫.PHP等领域 ...

  7. Java项目——物业管理系统(附源码+数据库)

    今天给小伙伴们分享一个Java项目--物业管理系统(附源码+数据库) 感兴趣的小伙伴可以点击下方链接和小编一起学习哟~ https://www.bilibili.com/video/BV1cD4y1s ...

  8. JAVA计算机毕业设计摄影网站(附源码、数据库)

    JAVA计算机毕业设计摄影网站(附源码.数据库) 目运行 环境项配置: Jdk1.8 + Tomcat8.5 + Mysql + HBuilderX(Webstorm也行)+ Eclispe(Inte ...

  9. nlu模型训练源码分析

    nlu模型训练源码分析 rasa/train.py是模型训练的文件,_train_async_internal函数是训练nlu和core的入口,_train_nlu_with_validated_da ...

最新文章

  1. android+sdk九宫格游戏,GitHub - mikegame/Android-SDKDemo: 米壳游戏Android SDKDemo
  2. 噪声dba是什么单位_在职DBA : 工作多年为什么还是选择报读工商管理博士
  3. 面试的算法1(C语言)(整理)(组合数 字符串倒置 最大公共串)
  4. 敲代码就是一把梭_2020必看!开发五年的大佬日常工作中所使用的java代码技巧...
  5. 【PC工具】200324更新百度网盘下载工具——最新百度网盘下载工具使用方法及注意事项...
  6. python xlwt写入excel_python xlwt模块写入excel超过65536行报错问题解决方法
  7. URAL 1427. SMS(DP+单调队列)
  8. Tricentis的测试工具评测
  9. UILocalNotification本地通知
  10. VxRack与VxRail遭受Power的“全面入侵”
  11. python2.7打印中文乱码的问题解决
  12. MATLAB怎么做出三叶玫瑰线,matlab复习题
  13. 我的 Java 血泪史
  14. matlab pn码捕获,直扩系统PN码捕获和跟踪的FPGA实现
  15. java saxreader,java – 如何使用dom4j SAXReader脱机?
  16. 阅读界面怎么用html做,如何在A4纸页面中制作HTML页面?
  17. 阿里云ddns ipk包下载
  18. 配电站房环境监测系统中使用的传感器
  19. 基于Seq2Seq的GRU时间序列预测Python程序
  20. Jenkins高级篇之Pipeline语法篇-2-第一个Pipeline脚本练习

热门文章

  1. 安然邮箱社交网络分析
  2. arecord录制音频和aplay播放音频用法说明
  3. 【Unity大气渲染】关于单次大气散射的理论知识
  4. CSS设置高度等于浏览器窗口
  5. 高位交叉和低位交叉_挑战408——组成原理(13)——并行存储器
  6. Tomcat 中文乱码解决过程
  7. TC-流量整形与限速
  8. 学会在Android Studio连接数据库
  9. 自主研发数据库TDSQL和TBase核心架构揭秘和实践
  10. 数据可视化 - 象形柱状图