一文读懂JVM类加载机制过程及原理

2023-10-28

在这里插入图片描述

一、做一个小测试,通过注释,标注出下面两个类中每个方法的执行顺序,并写出studentId的最终值。

package com.nezha.javase;

public class Person1 {

    private int personId;

    public Person1() {
        setId(100);
    }

    public void setId(int id) {
        personId = id;
    }
}
package com.nezha.javase;

public class Student1 extends Person1 {

    private int studentId = 1;

    public Student1() {
    }

    @Override
    public void setId(int id) {
        super.setId(id);
        studentId = id;
    }

    public void getStudentId() {
        System.out.println("studentId = " + studentId);
    }
}
package com.nezha.javase;

public class Test1 {
    public static void main(String[] args) {
        Student1 student = new Student1();
        System.out.println("new Student() 完毕,开始调用getStudentId()方法");
        student.getStudentId();
    }
}

有兴趣的小伙伴试一下,相信我,用System.out.println标记一下每个函数执行的先后顺序,如果你全对了,下面的不用看了,大佬。

在这里插入图片描述

二、类的初始化步骤:

  1. 初始化父类中的静态成员变量和静态代码块 ;
  2. 初始化子类中的静态成员变量和静态代码块 ;
  3. 初始化父类的普通成员变量和代码块,再执行父类的构造方法;
  4. 初始化子类的普通成员变量和代码块,再执行子类的构造方法;

三、看看你写对了没?

package com.nezha.javase;

public class Person {

    private int personId;

    /**
     * 第一步,走父类无参构造函数
     */
    public Person() {
        // 1、第一步,走父类无参构造函数
        System.out.println("第一步,走父类无参构造函数");
        System.out.println("");
        setId(100);
    }

    /**
     * 第三步,通过super.setId(id);走父类发方法
     * @param id
     */
    public void setId(int id) {
        System.out.println("第三步,通过super.setId(id);走父类发方法~~~id="+id);
        personId = id;
        System.out.println("在父类:studentId 被赋值为 " + personId);
        System.out.println("");
    }
}
package com.nezha.javase;

public class Student extends Person {

    private int studentId = 1;

    /**
     * 在走子类无参构造函数前,会先执行子类的普通成员变量初始化
     * 第五步,走子类无参构造函数
     */
    public Student() {
        System.out.println("第五步,在走子类无参构造函数前,会先执行子类的普通成员变量初始化");
        System.out.println("第六步,走子类无参构造函数");
        System.out.println("");
    }

    /**
     * 第二步,走子类方法
     *
     * 走完super.setId(id);,第四步,再回此方法
     * @param id
     */
    @Override
    public void setId(int id) {
        System.out.println("第二步,走子类方法~~id="+id);
        // 3、第三步,走子类方法
        super.setId(id);
        studentId = id;
        System.out.println("第四步,再回此方法,在子类:studentId 被赋值为 " + studentId);
        System.out.println("");
    }

    /**
     * 第六步,走getStudentId()
     */
    public void getStudentId() {
        // 4、打印出来的值是100
        System.out.println("第七步,走getStudentId()");
        System.out.println("studentId = " + studentId);
        System.out.println("");
    }
}
package com.nezha.javase;

public class Test1 {
    public static void main(String[] args) {
        Student1 student = new Student1();
        System.out.println("new Student() 完毕,开始调用getStudentId()方法");
        // 打印出来的值是100
        System.out.println("#推测~~打印出来的值是100");
        student.getStudentId();
    }
}

在这里插入图片描述

下面通过图解JVM的方式,分析一下。

四、类的加载过程

在这里插入图片描述

1、加载

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

2、链接

(1)验证(Verify)

  • 目的在于确保Class文件的字节流中包含信息符合当前虚拟机要求,保证被加载类的正确性,不会危害虚拟机自身安全;
  • 主要包括四种验证:文件格式验证、元数据验证、字节码验证、符号引用验证;

(2)准备(Prepare)

  • 为类变量分配内存并且设置该类变量的默认初始值;
  • 这里不包含final修饰的static,因为final在编译的时候就会分配了,准备阶段会显示初始化;
  • 这里不会为实例变量分配初始化,类变量会分配在方法区中,而实例变量是会随着对象一起分配到堆中;

(3)解析

  • 将常量池内的符号引用转换为直接引用的过程
  • 例如静态代码块、静态变量的显示赋值
  • 事实上,解析操作往往会伴随着JVM在执行完初始化之后在执行
  • 符号引用就是一组符号来描述所引用的目标。符号引用的字面量形式明确定义在《Java虚拟机规范》的Class文件格式中。直接引用就是指- 向目标的指针、相对偏移量或一个间接定位到目标的句柄
  • 解析动作主要针对类或接口、字段、类方法、接口方法、方法类型等。对常量池中的CONSTANT_Filedref_info、CONSTANT_Class_info、CONSTANT_Methodref_info等。

3、初始化

  • 初始化阶段就是执行类构造器方法的过程;
  • 此方法不需要定义,是javac编译器自动收集类中的所有类变量的赋值动作和静态代码块中的语句合并而来;
  • 构造器方法中指令按语句在源文件中出现的顺序执行;
  • 类构造器方法不同于类的构造器。构造器是虚拟机视角下的类构造器;
  • 若该类具有父类,JVM会保证子类的类构造器执行前,父类的类构造器已经执行完毕;
  • 虚拟机必须保证一个类的类构造器方法在多线程下被同步加锁;

五、类加载器的分类

JVM类加载器包括两种,分别为引导类加载器(Bootstrap ClassLoader)和自定义类加载器(User-Defined ClassLoader)。

所有派生于抽象类ClassLoader的类加载器划分为自定义类加载器。

1、启动类加载器(引导类加载器)

  1. 启动类加载器是使用C/C++语言实现的,嵌套在JVM内部;
  2. Java的核心类库都是使用引导类加载器加载的,比如String;
  3. 没有父加载器;
  4. 是扩展类加载器和应用程序类加载器的父类加载器 ;
  5. 出于安全考虑,Bootstrap启动类加载器只加载包名为java、javax、sun等开头的类 ;

在这里插入图片描述

2、扩展类加载器

  1. java语言编写
  2. 派生于ClassLoader类
  3. 父类加载器为启动类加载器
  4. 从java.ext.dirs系统属性所指定的目录中加载类库,或从JDK的安装目录jre/lib/ext子目录(扩展目录)下加载类库。如果用户创建的jar放在此目录下,也会自动由扩展类加载器加载

在这里插入图片描述

3、应用程序类加载器(系统类加载器)

  1. java语言编写
  2. 派生于ClassLoader类
  3. 父类加载器为扩展类加载器
  4. 它负责加载环境变量classpath或系统属性java.class.path指定路径下的类库
  5. 该类加载器是程序中默认的类加载器,一般来说,Java应用的类都是由它来完成加载的
  6. 通过ClassLoader.getSystemClassLoader()方法可以获得该类加载器

六、类加载器子系统的作用

在这里插入图片描述
类加载器子系统负责从文件系统或网络中加载class文件,class文件在文件开头有特定的文件标识。

ClassLoader只负责class文件的加载,至于它是否可以运行,则有执行引擎决定。

加载的类信息存放于一块称为方法区的内存空间。除了类的信息外,方法区中还会存放运行时常量池的信息,可能还包括字符串字面量和数字常量(这部分常量信息是class文件中常量池部分的内存映射)。

七、总结

类的初始化步骤,这看似非常基础的话题,却实打实的难住了很多人,还总结了更为深入JVM的类的加载过程、类加载器的分类、类加载器的作用。


在这里插入图片描述

Java学习路线总结,搬砖工逆袭Java架构师

10万字208道Java经典面试题总结(附答案)

Java基础教程系列

Java高并发编程系列

数据库进阶实战系列

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

一文读懂JVM类加载机制过程及原理 的相关文章

  • JVM 是否会内联对象的实例变量和方法?

    假设我有一个非常紧密的内部循环 每次迭代都会访问和改变一个簿记对象 该对象存储有关算法的一些简单数据 并具有用于操作它的简单逻辑 簿记对象是私有的和最终的 并且它的所有方法都是私有的 最终的和 inline 下面是一个示例 Scala 语法
  • 在 HP Load Runner 的 VuGen 中加载 javai.dll 时出现错误

    当我尝试在 HP load runner 的 VuGen 中编译一个简单的脚本时 无法启动 JVM 并出现以下错误 Java VM Internal Error Getting Error Loading javai dll 我在用着 HP
  • Java GuardedString - 用于加密的随机密钥是否存储在 Java 堆内存中?如果不是,那么密钥保存在哪里?

    Oracle 的 org identityconnectors common security GuardedString 要转换为 GuardedString 的原始数据需要由 EncryptorImpl class 随机生成的加密密钥
  • 强制jvm返回本机内存[重复]

    这个问题在这里已经有答案了 我时不时地运行需要大量内存的 eclipse 任务 因此 当任务运行时 jvm 会消耗大约 2 3GB 的 RAM 这是可以的 但是一旦 jvm 占用了该内存 它就不会释放它 并且我遇到了一种情况 堆中已用内存约
  • 在正在运行的 JVM 中查找正在运行的实例

    我想知道是否可以获取给定类的正在运行的实例的句柄 触发此问题的特定问题是应用程序由于存在大量正在运行的线程而无法正常退出 是的 我知道您可以对 thead 进行守护进程 然后它们就不会阻止应用程序退出 但这确实让我想知道这是否可能 我能做的
  • 调整 Java 类以提高 CPU 缓存友好性

    在设计java类时 对于实现CPU缓存友好性有哪些建议 到目前为止我学到的是应该尽可能多地使用 POD 即 int 而不是整数 这样 在分配包含对象时 数据将被连续分配 例如 class Local private int data0 pr
  • Java中的整数缓存[重复]

    这个问题在这里已经有答案了 可能的重复 奇怪的Java拳击 https stackoverflow com questions 3130311 weird java boxing 最近我看到一个演示 其中有以下 Java 代码示例 Inte
  • 使用 libjvm.so 时出现 Sigsegv Java 致命错误

    我正在做重启测试Sles12sp2 using STAF v3 4 24一段时间后我收到此错误 A fatal error has been detected by the Java Runtime Environment SIGSEGV
  • 使用 Coldfusion 11 的 CFdirectory,文件名中存在非 ASCII 字符问题

    我有一个类似的问题 ColdFusion CFDirectory 和法语 https stackoverflow com questions 1715632 coldfusion cfdirectory and the french从而没有
  • 如何在没有 Node.JS 的情况下运行 UglifyJS2

    无论如何都要跑UglifyJS2 https github com mishoo UglifyJS2没有node js 假设我想使用 JavaScript 脚本引擎在 JVM 进程中运行它 怎么做 我看到米秀回答你了https github
  • 将 Kotlin .kt 类打包到 JAR 中

    我如何构建HelloWorld kt as a JAR以便它运行 thufir dur kotlin thufir dur kotlin kotlinc HelloWorld kt include runtime d HelloWorld
  • 如何制作.Net或JVM语言?

    我看到了 NET 和 JVM 的所有这些新语言 一个人如何开始制作一个 我找不到关于 JVM 或 MSIL 规范的任何好的文档 Edit 我已经知道如何解析 我更感兴趣的是如何有这么多人基于这些平台创建新语言 你有点幸运 为 NET 开发的
  • Java 加载类时如何管理内存?

    想象一下 我有一个包含 10 个方法的类 我需要从该类中实例化 10 个对象 问题是 JVM 会在对象创建时为 10 个实例分配 10 个不同的内存空间 我的意思是在我调用构造函数时 即 new MyClass 吗 或者它会在内存中加载一次
  • 使用同一类的不同版本进行类加载:java.lang.LinkageError:尝试重复名称的类定义

    我有一个工作代码 可以动态加载具有不同类名的不同类实现 类文件被加载到内存数据库 Apache 德比数据库 类加载器检索 class文件来自BLOB列 我想做的是插入 class文件为带有版本列的二进制 BLOB 和IS ENABLED标志
  • Java 比 Xmx 参数消耗更多内存

    我有一个非常简单的 Web 服务器类 基于 Java SEHttpServer class 当我使用此命令启动编译的类来限制内存使用时 java Xmx5m Xss5m Xrs Xint Xbatch Test 现在如果我使用检查内存top
  • 使用 jni 从 C 调用 java 函数

    我正在编写一个简单的程序来从我的 C 程序调用 Java 函数 以下是我的代码 include
  • 从 Java 内部限制 CPU

    我在这个 和其他 论坛中看到了许多具有相同标题的问题 但似乎没有一个问题能完全解决我的问题 就是这个 我有一个 JVM 它占用了托管它的机器上的所有 CPU 我想限制它 但是我不能依赖任何限制工具 技术external到 Java 因为我无
  • 如果我使用最新的 JDK 编译 Java 文件,较旧的 JVM 是否能够运行 .class 文件?

    字节码是否取决于创建它所用的 Java 版本 如果我在最新的 JDK 中编译了 java 文件 较旧的 JVM 是否能够运行 class 文件 这取决于三件事 The actual Java versions you are talking
  • PS幸存者空间几乎已满

    我看到我的应用程序的 PS 幸存者空间在大部分时间几乎已满 98 我不知道PS幸存者空间是什么 这是正常的吗 遇到这种情况应该怎么办 首先 参见例如这里 什么是幸存者空间 https stackoverflow com q 10695298
  • Java:如何像 C++ 一样存储和检索内存地址

    我有 C 背景 在 C 中 我可以存储我刚刚在全局数组中新建的内存地址 并在以后重新使用它 例如 假设我有两个类 X Y 并且我创建了两个对象 x y 全局数组 StoreAddresses 2 定义为 uint32 t StoreAddr

随机推荐

  • 【IDEA Sprintboot】简单入门:整合SpringSecurity依赖、整合Thymeleaf框架

    目录 1 IDEA 简单入门 请求数据库表数据 水w的博客 CSDN博客 目录 三 1 整合SpringSecurity依赖 2 整合Thymeleaf框架 解决css样式等静态资源访问不到的问题 三 1 整合SpringSecurity依
  • Linux shell脚本之函数 Function 详解

    Linux shell脚本之函数Function 函数详解 函数语法 函数的生命周期 函数返回值 函数参数 变量作用域 1 本地变量 2 局部变量 函数变量示例 函数递归 递归示例 函数示例 函数详解 在过程式编程中 代码会重用 过程式编程
  • IOS集成ctp,恒生,金仕达穿透式监管的一点坑

    官方文档有些问题遗漏 1 要将引入对应头文件的文件用 mm后缀支持c 语法 不然会出现Expected 报错 不支持int 2 在解析的时候存在一些差异 CTP GetSystemInfo返回的数据实际是用char 接收的纯byte数据 用
  • 用RCircos包来画圈圈图

    用RCircos包来画圈圈图 我先讲解了画图 再讲解了一些小知识 安装并加载必须的packages 如果你还没有安装 就运行下面的代码安装 install packages RCircos library RCircos 如果你安装好了 就
  • GCC详解

    开放 自由和灵活是Linux的魅力所在 而这一点在gcc上的体现就是程序员通过它能够更好地控制整个编译过程 在使用gcc编译程序时 编译过程可以细分为 个阶段 预处理 Pre Processing 编译 Compiling 汇编 Assem
  • Linux命令大全(手册)

    https www linuxcool com
  • Android OkHttp源码阅读详解一

    博主前些天发现了一个巨牛的人工智能学习网站 通俗易懂 风趣幽默 忍不住也分享一下给大家 点击跳转到教程 前言 源码阅读基于okhttp 3 10 0 Android中OkHttp源码阅读二 责任链模式 implementation com
  • 【100天精通python】Day42:python网络爬虫开发_HTTP请求库requests 常用语法与实战

    目录 1 HTTP协议 2 HTTP与HTTPS 3 HTTP请求过程 3 1 HTTP请求过程 3 2 GET请求与POST请求 3 3
  • 树莓派快速扩容

    当树莓派默认的系统空间已经被占满 无法再进行安装等操作 而 SD卡仍有空间未被使用 可以对树莓派进行扩容 假设一个镜像的大小为 4G 由于该镜像对内存卡的要求是大于或等于 4G 但对于 4G以上的内存卡来说 树莓派的系统只利用到 4G 因此
  • 2.机器学习之单变量线性回归(李宏毅)

    2 单变量线性回归 Linear Regression with One Variable 2 1 模型表示 Model Represention 以前面说的例子 房子的价格是个回归问题 回归一词指的是 我们根据之前的数据预测出一个准确的输
  • 拦截SQLSERVER的SSL加密通道替换传输过程中的用户名密码实现运维审计(一)

    2019独角兽企业重金招聘Python工程师标准 gt gt gt 工作准备 一台SQLSERVER 2005 SQLSERVER 2008服务 SQLSERVER jdbc驱动程序 Java开发环境eclipse jdk1 8 java反
  • java爬虫代码示例_Excel VBA 实战(8) - 巨潮资讯 VBA 爬虫

    2019年6月25日 更新 由于巨潮资讯查询API变更 此文章当中的所涉及代码也进行了相应更新 虽然原代码当前已经失效 但是爬虫的基本逻辑仍然适用 因此请没有相关基础的朋友们先行阅读此文 实战代码部分请移步至如下链接 杨风飒 Excel V
  • Spring data Jpa操作ES

    以下是使用Spring data Jpa操作ES的一些记录 在ElasticsearchRepository中我们可以使用Not Add Like Or Between等关键词自动创建查询语句 记住上面这句话 代码示例 public int
  • 【Linux】进程信号 -- 信号产生

    信号的旧识引入 信号引入 signal调用 系统调用向目标进程发送信号 模拟实现一个kill命令 raise给自己发送任意信号 abort给自己发送指定信号 6 SIGABRT 硬件异常产生信号 除0异常 野指针访问异常 软件条件产生信号
  • git: ‘remote-codecomit‘ is not a git command. See ‘git --help‘.

    git remote codecomit is not a git command See git help 是什么情况出现该错误 重复安装python后出现 做过的尝试 在谷歌百度各大网站上找寻没有发现跟我类似的情况 然后就是尝试重新安装
  • MQTT、CoAP 还是 LwM2M?主流物联网协议如何选择

    随着物联网技术的发展与普及 越来越多的智能设备具备了网络连接与数据传输能力 由于物联网场景复杂多样 设备端硬件条件 网络稳定性 流量限制 设备功耗以及设备连接数量等多方面因素造成物联网设备的消息传递与传统互联网场景有着很大不同 也因此产生了
  • SQLi LABS Less-35

    第三十五关注入点为 数值型 注入方式为 报错注入 此关卡通过 代码WAF 转义了单引号 我们使用 编译 绕过WAF 先上结果 id 1 and updatexml 1 concat 0x7e substr select group conc
  • qaz69.com forum.php,Eclipse Community Forums

    Originally posted by richkulp us NO SPAM ibm com This is a multi part message in MIME format 080707010607010307000006 Co
  • 尝试简单的Spigot插件开发, PowerItemCraft 武器增强插件 (Spigot开发笔记-2)

    文章目录 创建项目 插件功能蓝图 实现思路 1 指令模块 2 事件模块 3 插件中间层 3 1 主类的工作 3 2 如何解耦 完整插件 创建项目 根据 Spigot开发笔记 1 中的内容创建好项目 插件功能蓝图 匠魂mod中有一个随着使用会
  • 一文读懂JVM类加载机制过程及原理

    目录 一 做一个小测试 通过注释 标注出下面两个类中每个方法的执行顺序 并写出studentId的最终值 二 类的初始化步骤 三 看看你写对了没 四 类的加载过程 1 加载 2 链接 3 初始化 五 类加载器的分类 1 启动类加载器 引导类