java八股文面试[JVM]——双亲委派模型

1.当AppClassLoader 去加载一个class时,它首先不会自己去尝试加载这个类,而是把类加载请求委托给父加载器ExtClassLoader 去完成。

2.当ExtClassLoader去加载一个class时,它首先也不会去尝试加载这个类,而是把类加载请求委托给父加载器BootstrapClassLoader 去完成。

3.如果BootstrapClass加载失败(例如在JAVA_HOME/jre/lib里未找到该class),就会使用ExtClassLoader来尝试加载。

4.如果使用ExtClassLoader加载失败(例如在JAVA_HOME/jre/lib/ext里未找到该class),就会使用AppClassLoader来尝试加载。如果AppClassLoader也加载失败,则会抛出异常ClassNotFoundException

从代码层面了解几个Java中定义的类加载器及双亲委派模型的实现,他们的类图关系如下。

从图可以看出顶层的类加载器是抽象类abstract ClassLoader,其后所有的类加载器都继承自ClassLoader(不包括启动类加载器)。为了更好的理解双亲委派模型,ClassLoader源码中的loadClass(String)方法该方法加载指定名称(包括包名)的二进制类型,该方法在jdk1.2后不建议用户重写但是可以直接调用该方法。loadClass ()方法是ClassLoader类自己实现的,该方法中的逻辑就是双亲委派模式的实现loadClass(String name, boolean resolve)是一个重载方法,resolve参数代表是否生成Class对象的进行解析相关的操作,源码分析如下。

双亲委派模型意义总结来讲就是:

1.系统类防止内存中出现多份相同的字节码。

2.保证Java程序安全稳定运行。

加载类的三种方式

到这里,相信大家都对类加载器和类的加载都有一定的了解了,那么你知道嘛?常见的加载类的方式有三种。

  1. 静态加载:也就是使用new关键字来创建实例对象,
  2. 动态加载:使用Class.forName()动态加载(反射加载类型),然后调用类的newInstance()方法实例化对象。
  3. 动态加载:通过类加载器的loadClass()方法来加载类,然后调用类的newInstance()方法实例化对象。
三种方式的区别
  1. 第一种和第二种 方式使用的类加载器是相同的,都是当前类加载器(this.getClass().getClassLoader ()),第三种方式需要用户指定类加载器。
  2. 如果需要在当前类路径以外加载类,则只能使用第三种方式,第三种方式加载的类与当前类分属不同的命名空间
  3. 第一种方式是静态加载,第二种和第三种是动态加载。
两种异常(Exception)
  1. 静态加载的时候如果在运行环境中找不到要初始化的类,抛出的是NoClassDefFoundError。它在Java的异常体系中是一个**error**。
  2. 动态加载的时候如果运行环境中找不到要初始化的类,抛出的是ClassNotFoundException。它在Java的异常体系中是一个**checked异常**。
Class.forName与ClassLoader.loadClass的区别

首先,我们必须要明白类加载机制的三个主要过程是:加载-->连接-->初始化。

  • Class.forName():除了将类的.class文件加载到JVM中之外,还对类进行解释 ,执行类中的static块。
  • ClassLoder.loadClass():只是 将类的.class文件加载到JVM中,并不会执行类的static块,只有在newInstance才会执行static块
  • Class.forName(name, initialize, loader):带参函数也可控制是否加载static块。并且只有调用了newInstance()方法采用调用构造函数。创建类的对象 。

下面我们写下代码,加深一下影响。

java 复制代码
public class Demo {
    static {
        System.out.println("正在加载Demo...");
    }
}
java 复制代码
import java.lang.Class;

public class TestDemo {
    public static void main(String[] args) throws ClassNotFoundException {
        //1.使用 Class.forName()加载类,默认执行static块。
        Class.forName("Demo");
        //获得当前类的类加载器
        ClassLoader classLoader = Test.class.getClassLoader();
        //2.使用 Class.forName()加载类,并指定ClassLoader。初始化时不会执行static块。
        Class.forName("Demo", false, classLoader);
        //3.使用ClassLoader类的loadClass()方法加载类,不会执行初始化
        classLoader.loadClass("Demo");
    }
}

只有第一行代码输出了

知识来源:

【基础】双亲委派模型_哔哩哔哩_bilibili

深入理解Java类加载器(ClassLoader)

相关推荐
程序猿小D19 分钟前
[附源码+数据库+毕业论文]基于Spring+MyBatis+MySQL+Maven+jsp实现的个人财务管理系统,推荐!
java·数据库·mysql·spring·毕业论文·ssm框架·个人财务管理系统
zhuiQiuMX32 分钟前
脉脉maimai面试死亡日记
数据仓库·sql·面试
独行soc36 分钟前
2025年渗透测试面试题总结-2025年HW(护网面试) 33(题目+回答)
linux·科技·安全·网络安全·面试·职场和发展·护网
库森学长1 小时前
面试官:发生OOM后,JVM还能运行吗?
jvm·后端·面试
转转技术团队1 小时前
二奢仓店的静默打印代理实现
java·后端
然我1 小时前
面试必问:JS 事件机制从绑定到委托,一篇吃透所有考点
前端·javascript·面试
钢铁男儿1 小时前
C# 接口(什么是接口)
java·数据库·c#
丶小鱼丶1 小时前
排序算法之【归并排序】
java·排序算法
上上迁2 小时前
分布式生成 ID 策略的演进和最佳实践,含springBoot 实现(Java版本)
java·spring boot·分布式
永日456702 小时前
学习日记-spring-day42-7.7
java·学习·spring