JVM是如何加载类的?

简介: Java虚拟机加载类的全过程包括,加载,验证,准备,解析和初始化。image.png在加载阶段,虚拟机需要完成以下三件事:通过类的全限名获取此类的二进制字节流。

Java虚拟机加载类的全过程包括,加载,验证,准备,解析和初始化。


img_d85629228e38fa380afe29ea61a94103.png
image.png

在加载阶段,虚拟机需要完成以下三件事:

  1. 通过类的全限名获取此类的二进制字节流。
  2. 将这个字节流所代表的静态存储结构转换为方法区的运行时数据区
  3. 在内存中生成一个代表这个类的Class对象,作为方法区的这个类的各种数据访问入口。

可以看出,Java能通过加载外部的字节码来实现动态的装载类,这为Java提供了很大的灵活性。实现类加载动作的代码叫做类加载器。

比较两个类是否相等,只有在两个类是由同一个类加载器加载的前提下才有意义,否则尽管两个类是同一个Class文件,只要类加载器不同,那么这两个类必定不相等。

双亲委派机制

绝大部分Java程序都会用到以下三种系统提供的类加载器:

  • BootStrap ClassLoader,负责加载<JAVA_HOME>/lib或被-Xbootclasspath指定路径下的类库,开发者不可以直接使用
  • Extension ClassLoader,负责加载<JAVA_HOME>/lib/ext或被java.ext.dirs系统变量指定的路径中的所有类库,开发者可以直接使用
  • App ClassLoader,这个类加载器是ClassLoader.getSystemClassLoader()的返回值,负责加载用户类路径上所指定的类库,开发者可以直接使用这个类加载器,如果应用程序没有自定义过类加载器,那么系统默认使用这个类加载器。

应用程序一般都会用到以上三种类加载器,如果有必要我们可以指定自己的类加载器。

img_4432910d978dc4d30fb3f9b51c4ca14b.png
image.png

图中展示的这种层次关系,称为类加载的双亲委派模型,除了顶层的启动类加载器之外,其余的类加载都要有自己的父类加载器。

类加载器之间的关系不是以继承的方式实现,而是以组合的方式实现。

工作原理

双亲委派的工作流程:如果一个类收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把请求委派给父类来实现,每一个层次的类加载器都是这样,因此所有的类加载请求都会最终传送到启动类加载器,只有当父类加载器无法完成这个加载请求,子类加载器才会自己尝试加载。

要点:

  • 类加载请求全部交给自己的父类来操作
  • 父类加载器加载不了的自己加载
/**

*/
    protected Class<?> loadClass(String name, boolean resolve)
        throws ClassNotFoundException
    {
        synchronized (getClassLoadingLock(name)) {
            // 首先检查,类是否已经被加载
            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) {
                    //父类加载器无法加载类,抛出异常
                }
                
                //如果父类没有加载成功,然后自己寻找对应的类, 我们可以实现自己的findClass,进而实现自定义类加载器
                if (c == null) {
                    long t1 = System.nanoTime();
                    c = findClass(name);                
//记录状态
                    sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
                    sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
                    sun.misc.PerfCounter.getFindClasses().increment();
                }
            }
            if (resolve) {
                resolveClass(c);
            }
            return c;
        }
    }

双亲委派机制一个好处:避免内存中出现同样的字节码。可以很好的解决各个类加载器的基础类统一问题。

我个人理解,所有的类最终是由启动类加载器加载的,而启动类加载器我们是不可以直接使用的,启动类加载器的代码由大神写的,肯定比我们的更安全。常见的系统级别的类都由启动类加载器加载也保证了安全,不然如果系统级别的类,由我们写的类加载器加载,这样多个系统类出现不一致的情况,让语言变得很不稳定。

破坏双亲委派机制

双亲委派机制是一种推荐的使用方式,但不是强制的,虽然绝大部分Java应用都是使用双亲委派模型,但是也有例外。比如热替换,热部署。

OSGI是Java业界广泛认可的模块化标准,而OSGI模块化热部署的关键是它自定义的类加载器。每一个模块都有一个自己的类加载器,当需要更换一个 Bundle(包) 时,Bundle连同类加载器一同替换实现代码热部署。

弄懂了OSGi的精髓,就可以算是掌握了类加载器的精髓

大家一起加油。

最后

本文介绍了类加载中的加载过程,其中加载过程由类加载器来完成,类加载器的加载利用到了双亲委派机制,通过代码,我们可以更好的理解双亲委派机制,我们也知道了双亲委派机制不是要强制实现的,可以试着破坏双亲委派机制,重新loadClass方法即可..,自定义类加载器需要重写的是findClass方法。

希望能帮到大家

参考

  • 《深入理解JVM》
相关文章
|
13天前
|
存储 安全 Java
JVM加载过程
JVM类加载过程是Java开发中的关键环节,主要包括五个阶段:加载、验证、准备、解析和初始化。加载阶段获取类的二进制字节流;验证确保字节码符合规范;准备为静态变量分配内存并默认初始化;解析将符号引用转为直接引用;初始化执行静态变量赋值和静态代码块。了解这一过程有助于深入理解Java程序运行机制,提升编程水平。
|
4月前
|
安全 Java 应用服务中间件
JVM常见面试题(三):类加载器,双亲委派模型,类装载的执行过程
什么是类加载器,类加载器有哪些;什么是双亲委派模型,JVM为什么采用双亲委派机制,打破双亲委派机制;类装载的执行过程
110 35
JVM常见面试题(三):类加载器,双亲委派模型,类装载的执行过程
|
3月前
|
缓存 前端开发 Java
JVM知识体系学习二:ClassLoader 类加载器、类加载器层次、类过载过程之双亲委派机制、类加载范围、自定义类加载器、编译器、懒加载模式、打破双亲委派机制
这篇文章详细介绍了JVM中ClassLoader的工作原理,包括类加载器的层次结构、双亲委派机制、类加载过程、自定义类加载器的实现,以及如何打破双亲委派机制来实现热部署等功能。
86 3
|
3月前
|
小程序 Oracle Java
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
这篇文章是关于JVM基础知识的介绍,包括JVM的跨平台和跨语言特性、Class文件格式的详细解析,以及如何使用javap和jclasslib工具来分析Class文件。
62 0
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
|
7月前
|
Java 编译器
Java健壮性 Java可移植性 JDK, JRE, JVM三者关系 Java的加载与执行原理 javac编译与JAVA_HOME环境变量介绍 Java中的注释与缩进 main方法的args参数
Java健壮性 Java可移植性 JDK, JRE, JVM三者关系 Java的加载与执行原理 javac编译与JAVA_HOME环境变量介绍 Java中的注释与缩进 main方法的args参数
67 1
|
6月前
|
Java Perl
JVM内存问题之如何统计在JVM的类加载中,每一个类的实例数量,并按照数量降序排列
JVM内存问题之如何统计在JVM的类加载中,每一个类的实例数量,并按照数量降序排列
|
6月前
|
存储 安全 Java
开发与运维引用问题之JVM类加载过程如何解决
开发与运维引用问题之JVM类加载过程如何解决
37 0
|
6月前
|
存储 算法 Java
JAVA程序运行问题之Java类加载到JVM中加载类时,实际上加载的是什么如何解决
JAVA程序运行问题之Java类加载到JVM中加载类时,实际上加载的是什么如何解决
|
7月前
|
安全 前端开发 Java
《JVM由浅入深学习【一】 》JVM由简入深学习提升(类加载过程+父子类加载过程+类加载器+双亲委派机制)
《JVM由浅入深学习【一】 》JVM由简入深学习提升(类加载过程+父子类加载过程+类加载器+双亲委派机制)
54 0
|
7月前
|
存储 Java 编译器
【搞定Jvm面试】 面试官:谈谈 JVM 类文件结构的认识
【搞定Jvm面试】 面试官:谈谈 JVM 类文件结构的认识