欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 新闻 > 国际 > JVM 加载 class 文件的原理机制

JVM 加载 class 文件的原理机制

2024/10/24 4:58:59 来源:https://blog.csdn.net/2301_77207909/article/details/143181464  浏览:    关键词:JVM 加载 class 文件的原理机制

JVM 加载 class 文件的原理机制

JVM(Java虚拟机)是一个可以执行Java字节码的虚拟机。它负责执行Java应用程序和应用程序的扩展,如Java库和框架。

文章目录

  • JVM 加载 class 文件的原理机制
    • 1. JVM
      • 1.1 类加载器
      • 1.2 魔数
      • 1.3 元空间
    • 2. 类加载
      • 2.1 类加载过程
      • 2.2 双亲委派机制
        • 模型原理
        • 模型优势
        • 模型实现

1. JVM

1.1 类加载器

类加载器是JVM用来查找和加载.class文件到JVM中的组件。类加载器的主要职责是从指定的位置找到.class文件,然后将其读入到内存中,并生成对应的java.lang.Class对象。

  • 启动类加载器(Bootstrap ClassLoader):用原生代码实现,负责加载JVM核心库中的.class文件,如rt.jar中的类。
  • 扩展类加载器(Extension ClassLoader):由Java实现,负责加载jre/lib/ext目录中的.class文件或由系统变量java.ext.dirs指定的目录中的.class文件。
  • 应用程序类加载器(Application ClassLoader):由Java实现,负责加载用户类路径(classpath)中的.class文件。
  • 用户自定义类加载器:可以由用户自己编写,继承自java.lang.ClassLoader,用于特殊目的的类加载。

1.2 魔数

.class文件的开头四个字节被称为魔数(Magic Number)。魔数是0xCAFEBABE,它是用来识别一个文件是否是.class文件。如果文件不是.class文件,或者魔数不匹配,JVM将无法执行该文件。
在这里插入图片描述

1.3 元空间

在JVM中,.class文件加载后,类信息被存储在方法区中,这部分内存区域被称为元空间(Metaspace)。元空间是方法区的一部分,用于存储类元数据,包括类的定义信息、静态变量、常量池等。

2. 类加载

2.1 类加载过程

类加载过程大致可以分为以下几个步骤

  1. 加载(Loading):找到.class文件,并将其读入内存,创建一个java.lang.Class对象。
  2. 链接(Linking):验证.class文件的正确性,准备类在JVM中运行所需的内存,并解析符号引用。
  3. 初始化(Initialization):执行类构造器方法<clinit>(),初始化类变量和静态初始化块。

2.2 双亲委派机制

双亲委派模型(Parent Delegation Model)是Java虚拟机(JVM)中类加载机制的核心设计之一。它定义了类加载器之间的加载顺序和委托规则,确保了类加载过程的一致性和安全性。
在这里插入图片描述

模型原理

在双亲委派模型中,类加载器分为层次结构,子类加载器会首先请求其父类加载器完成类的加载任务。这个过程遵循:

  1. 当一个类加载器需要加载某个类时,它会首先请求其父类加载器加载该类。
  2. 如果父类加载器能够成功加载该类,则直接返回这个类;如果父类加载器无法加载,则子类加载器会尝试自己加载该类。
  3. 如果子类加载器也无法加载该类,则会抛出ClassNotFoundException异常。

这种委托关系的链条从下往上,一直到达启动类加载器(Bootstrap ClassLoader),它是所有类加载器的顶层,负责加载JVM的核心库(如rt.jar中的类)。

模型优势
  • 避免类的重复加载:由于类加载器之间的委托关系,同一类只会在父类加载器中加载一次,避免了重复加载同一个类,节省了资源。
  • 保证类型安全:所有非启动类加载器都委托给启动类加载器加载核心库中的类,确保了这些类都是由可信的类加载器加载的,从而保证了类型安全。
  • 防止核心API被篡改:核心库的类由启动类加载器加载,任何试图替换这些类的尝试都会被父类加载器拦截,从而保护了Java核心API不被篡改。
模型实现

双亲委派模型的实现依赖于Java的ClassLoader类及其子类。

public class ClassLoader {private ClassLoader parent;public ClassLoader(ClassLoader parent) {this.parent = parent;}public Class<?> loadClass(String name) throws ClassNotFoundException {// 首先检查是否已经加载过该类Class<?> c = findLoadedClass(name);if (c == null) {try {// 如果父类加载器不为空,则委托父类加载器if (parent != null) {c = parent.loadClass(name);} else {// 如果父类加载器为空,则由Bootstrap ClassLoader加载c = findBootstrapClass(name);}} catch (ClassNotFoundException e) {// 如果父类加载器加载失败,则由当前类加载器加载c = findClass(name);}}return c;}// ... 其他方法 ...
}

ClassLoaderloadClass方法首先尝试从缓存中查找已经加载的类,如果没有找到,则根据双亲委派模型委托给父类加载器加载。如果父类加载器也无法加载,则由当前类加载器负责加载。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com