JVM_类的加载、链接、初始化、卸载、主动使用、被动使用

news2025/1/30 10:38:08

①. 说说类加载分几步?

  • ①. 按照Java虚拟机规范,从class文件到加载到内存中的类,到类卸载出内存为止,它的整个生命周期包括如下7个阶段:

  1. 第一过程的加载(loading)也称为装载
  2. 验证、准备、解析3个部分统称为链接(Linking)
  3. 在Java中数据类型分为基本数据类型和引用数据类型。基本数据类型由虚拟机预先定义,引用数据类型则需要进行类的加载

②. 当程序要使用某个类时,如果该类还未被加载到内存中,则系统会通过类的加载、类的链接、类的初始化这三个步骤来对类进行初始化。如果不出现意外,JVM将会连续完成这三个步骤,所以有时也把这三个步骤统称为类加载或者初始化

③. 从程序中类的使用过程看:

    ②. 过程一:类的加载(Loading)

    • ①. 类的加载指的是将类的.class文件中的二进制数据读取到内存中,存放在运行时数据区的方法区中,并创建一个大的Java.lang.Class对象,用来封装方法区内的数据结构 在加载类时,Java虚拟机必须完成以下3件事情:

    ②. 对于①的说明,我们也可以这样去理解:所谓装载(加载),简而言之就是将Java类的字节码文件加载到机器内存中,并在内存中构建出Java类的原型——类模板对象

    (所谓类模板对象,其实就是Java类在JVM内存中的一个快照,JVM将从字节码文件中解析出的常量池、类字段、类方法等信息存储到类模板中,这样JVM在运行期便能通过类模板而获取Java类中的任意信息,能够对Java类的成员变量进行遍历,也能进行Java方法的调用)

    ③. 对于类的二进制数据流,虚拟机可以通过多种途径产生或获得(只要所读取的字节码符合JVM规范即可)

    1.虚拟机可能通过文件系统读入一个class后缀的文件(最常见)

    2.读入jar、zip等归档数据包,提取类文件。

    3.事先存放在数据库中的类的二进制数据

    4.使用类似于HTTP之类的协议通过网络进行加载

    5.在运行时生成一段Class的二进制信息等

    ④. Class实例的位置

    (类将.class文件加载至元空间后,会在堆中创建一个Java.lang.Class对象,用来封装类位于方法区内的数据结构,该Class对象是在加载类的过程中创建的,每个类都对应有一个Class类型的对象)

    1. 通过类的全名,获取类的二进制数据流
    2. 解析类的二进制数据流为方法区内的数据结构(Java类模型)
    3. 创建java.lang.Class类的实例,表示该类型。作为方法区这个类的各种数据的访问入口

    ③. 过程二:链接(Linking)

    • ①. 验证:确保Class文件的字节流中包含信息符合当前虚拟机要求,保证被加载类的正确性

      1.  

      1.目的是确保Class文件的字节流中包含信息符合当前虚拟机要求,保证被加载类的正确性,不会危害虚拟机自身安全
      2.主要包括四种验证:文件格式验证,元数据验证,字节码验证,符号引用验证
      3.格式检查:是否以魔术oxCAFEBABE开头,主版本和副版本是否在当前Java虚拟机的支持范围内,数据中每一项是否都拥有正确的长度等

        ②. 准备(静态变量,不能是常量)

          1.为类变量分配内存并且设置该类变量的默认初始化值
          2.这里不包含用final修饰的static,因为final在编译的时候就会分配了,准备阶段会显式赋值
          3.这里不会为实例变量分配初始化,类变量会分配在方法区中,而实例变量会随着对象一起分配到Java堆中
          4.注意:Java并不支持boolean类型,对于boolean类型,内部实现是int,由于int的默认值是0,故对应的,boolean的默认值就是false

            public class ClassInitTest {
                private  static int num=1; //类变量的赋值动作
                //静态代码快中的语句
                static{
                    num=2;
                    number=20;
                    System.out.println(num);
                    //System.out.println(number); 报错:非法的前向引用
                }
                //Linking之prepare: number=0 -->initial:20-->10
                private static int number=10;
            
                public static void main(String[] args) {
                    System.out.println(ClassInitTest.num);
                    System.out.println(ClassInitTest.number);
                }
            }
            

          • ③. 若该类具有父类,Jvm会保证子类的< clinit >() 执行前,父类的< clinit >() 已经执行完成。clinit 不同于类的构造方法(init) (由父及子,静态先行)

           

          public class ClinitTest1 {
              static class Father{
                  public static int A=1;
                  static{
                      A=2;
                  }
              }
              static class Son extends Father{
                  public static int B=A;
              }
          
              public static void main(String[] args) {
                  //这个输出2,则说明父类已经全部加载完毕
                  System.out.println(Son.B);
              }
          }
          

           

          /**
           * @author TANGZHI
           * @create 2021-01-01 18:49
           * 哪些场景下,java编译器就不会生成<clinit>()方法
           */
          public class InitializationTest1 {
              //场景1:对应非静态的字段,不管是否进行了显式赋值,都不会生成<clinit>()方法
              public int num = 1;
              //场景2:静态的字段,没有显式的赋值,不会生成<clinit>()方法
              public static int num1;
              //场景3:比如对于声明为static final的基本数据类型的字段,不管是否进行了显式赋值,都不会生成<clinit>()方法
              public static final int num2 = 1;
          }
          

           

          
          /**
           * @author TANGZHI
           * @create 2021-01-01 
           *
           * 说明:使用static + final修饰的字段的显式赋值的操作,到底是在哪个阶段进行的赋值?
           * 情况1:在链接阶段的准备环节赋值
           * 情况2:在初始化阶段<clinit>()中赋值
           * 结论:
           * 在链接阶段的准备环节赋值的情况:
           * 1. 对于基本数据类型的字段来说,如果使用static final修饰,则显式赋值(直接赋值常量,而非调用方法)通常是在链接阶段的准备环节进行
           * 2. 对于String来说,如果使用字面量的方式赋值,使用static final修饰的话,则显式赋值通常是在链接阶段的准备环节进行
           *
           * 在初始化阶段<clinit>()中赋值的情况:
           * 排除上述的在准备环节赋值的情况之外的情况。
           * 最终结论:使用static + final修饰,且显示赋值中不涉及到方法或构造器调用的基本数据类型或String类型的显式赋值,是在链接阶段的准备环节进行。
           */
          public class InitializationTest2 {
              public static int a = 1;//在初始化阶段<clinit>()中赋值
              public static final int INT_CONSTANT = 10;//在链接阶段的准备环节赋值
          
              public static final Integer INTEGER_CONSTANT1 = Integer.valueOf(100);//在初始化阶段<clinit>()中赋值
              public static Integer INTEGER_CONSTANT2 = Integer.valueOf(1000);//在初始化阶段<clinit>()中赋值
          
              public static final String s0 = "helloworld0";//在链接阶段的准备环节赋值
              public static final String s1 = new String("helloworld1");//在初始化阶段<clinit>()中赋值
          
              public static String s2 = "helloworld2";
              public static final int NUM1 = new Random().nextInt(10);//在初始化阶段<clinit>()中赋值
          }
          

           

          package com.xiaozhi;
          
          /**
           * @author TANGZHI
           * @create 2021-05-25
           */
          class StaticA {
              static {
                  try {
                      Thread.sleep(1000);
                  } catch (InterruptedException e) {
                  }
                  try {
                      Class.forName("com.xiaozhi.StaticB");
                  } catch (ClassNotFoundException e) {
                      e.printStackTrace();
                  }
                  System.out.println("StaticA init OK");
              }
          }
          
          class StaticB {
              static {
                  try {
                      Thread.sleep(1000);
                  } catch (InterruptedException e) {
                  }
                  try {
                      Class.forName("com.xiaozhi.StaticA");
                  } catch (ClassNotFoundException e) {
                      e.printStackTrace();
                  }
                  System.out.println("StaticB init OK");
              }
          }
          
          public class StaticDeadLockMain extends Thread {
              private char flag;
          
              public StaticDeadLockMain(char flag) {
                  this.flag = flag;
                  this.setName("Thread" + flag);
              }
          
              @Override
              public void run() {
                  try {
                      Class.forName("com.xiaozhi.Static" + flag);
                  } catch (ClassNotFoundException e) {
                      e.printStackTrace();
                  }
                  System.out.println(getName() + " over");
              }
          
              public static void main(String[] args) throws InterruptedException {
                  StaticDeadLockMain loadA = new StaticDeadLockMain('A');
                  loadA.start();
                  StaticDeadLockMain loadB = new StaticDeadLockMain('B');
                  loadB.start();
              }
          }
          

          ⑤. 主动引用(触发在初始化阶段的Clinit方法)

             

            # 注意,如果把A接口中的默认方法注释,那么就只输出:子类初始化......
            输出:
            CompareB的初始化
            子类初始化.....
            public class DemoB implements A{
                static{
                    System.out.println("子类初始化......");
                }
                public static void main(String[] args) {

                }
            }
            interface A{
                public static final Thread t = new Thread() {
                    {
                        System.out.println("CompareB的初始化");
                    }
                };
                default void method1(){
                    System.out.println("====");
                }
            }
             

            ⑥. 被动使用

            ⑦. 过程四:类的Using(使用)

            ⑧. 过程五:类的Unloading(卸载)

                      本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2286384.html

                      如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

                      相关文章

                      ProfibusDP主机与从机交互

                      ProfibusDP 主机SD2索要数据下发&#xff1a;68 08 F7 68 01 02 03 21 05 06 07 08 1C 1668&#xff1a;SD2 08&#xff1a;LE F7&#xff1a;LEr 68&#xff1a;SD2 01:目的地址 02&#xff1a;源地址 03:FC_CYCLIC_DATA_EXCHANGE功能码 21&#xff1a;数据地址 05,06,07,08&a…

                      Java设计模式:结构型模式→组合模式

                      Java 组合模式详解 1. 定义 组合模式&#xff08;Composite Pattern&#xff09;是一种结构型设计模式&#xff0c;它允许将对象组合成树形结构以表示“部分-整体”的层次。组合模式使得客户端能够以统一的方式对待单个对象和对象集合的一致性&#xff0c;有助于处理树形结构…

                      【福州市AOI小区面】shp数据学校大厦商场等占地范围面数据内容测评

                      AOI城区小区面样图和数据范围查看&#xff1a; — 字段里面有name字段。分类比较多tpye&#xff1a;每个值代表一个类型。比如字段type中1549代表小区住宅&#xff0c;1563代表学校。小区、学校等占地面积范围数据 —— 小区范围占地面积面数据shp格式 无偏移坐标&#xff0c;只…

                      【Python实现机器遗忘算法】复现2023年TNNLS期刊算法UNSIR

                      【Python实现机器遗忘算法】复现2023年TNNLS期刊算法UNSIR 1 算法原理 Tarun A K, Chundawat V S, Mandal M, et al. Fast yet effective machine unlearning[J]. IEEE Transactions on Neural Networks and Learning Systems, 2023. 本文提出了一种名为 UNSIR&#xff08;Un…

                      基于SpringBoot的阳光幼儿园管理系统

                      作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、Vue项目源码、SSM项目源码、微信小程序源码 精品专栏&#xff1a;…

                      【开源免费】基于SpringBoot+Vue.JS景区民宿预约系统(JAVA毕业设计)

                      本文项目编号 T 162 &#xff0c;文末自助获取源码 \color{red}{T162&#xff0c;文末自助获取源码} T162&#xff0c;文末自助获取源码 目录 一、系统介绍二、数据库设计三、配套教程3.1 启动教程3.2 讲解视频3.3 二次开发教程 四、功能截图五、文案资料5.1 选题背景5.2 国内…

                      安卓逆向之脱壳-认识一下动态加载 双亲委派(一)

                      安卓逆向和脱壳是安全研究、漏洞挖掘、恶意软件分析等领域的重要环节。脱壳&#xff08;unpacking&#xff09;指的是去除应用程序中加固或保护措施的过程&#xff0c;使得可以访问应用程序的原始代码或者数据。脱壳的重要性&#xff1a; 分析恶意软件&#xff1a;很多恶意软件…

                      马尔科夫模型和隐马尔科夫模型区别

                      我用一个天气预报和海藻湿度观测的比喻来解释&#xff0c;保证你秒懂&#xff01; 1. 马尔可夫模型&#xff08;Markov Model, MM&#xff09; 特点&#xff1a;状态直接可见 场景&#xff1a;天气预报&#xff08;晴天→雨天→阴天…&#xff09;核心假设&#xff1a; 下一个…

                      Python NumPy(7):连接数组、分割数组、数组元素的添加与删除

                      1 连接数组 函数描述concatenate连接沿现有轴的数组序列stack沿着新的轴加入一系列数组。hstack水平堆叠序列中的数组&#xff08;列方向&#xff09;vstack竖直堆叠序列中的数组&#xff08;行方向&#xff09; 1.1 numpy.concatenate numpy.concatenate 函数用于沿指定轴连…

                      【LLM】deepseek多模态之Janus-Pro和JanusFlow框架

                      note 文章目录 note一、Janus-Pro&#xff1a;解耦视觉编码&#xff0c;实现多模态高效统一技术亮点模型细节 二、JanusFlow&#xff1a;融合生成流与语言模型&#xff0c;重新定义多模态技术亮点模型细节 Reference 一、Janus-Pro&#xff1a;解耦视觉编码&#xff0c;实现多模…

                      2000-2021年 全国各地级市专利申请与获得情况、绿色专利申请与获得情况数据

                      2000-2021年 全国各地级市专利申请与获得情况、绿色专利申请与获得情况数据.ziphttps://download.csdn.net/download/2401_84585615/89575931 https://download.csdn.net/download/2401_84585615/89575931 2000至2021年&#xff0c;全国各地级市的专利申请与获得情况呈现出显著…

                      51单片机(STC89C52)开发:点亮一个小灯

                      软件安装&#xff1a; 安装开发板CH340驱动。 安装KEILC51开发软件&#xff1a;C51V901.exe。 下载软件&#xff1a;PZ-ISP.exe 创建项目&#xff1a; 新建main.c 将main.c加入至项目中&#xff1a; main.c:点亮一个小灯 #include "reg52.h"sbit LED1P2^0; //P2的…

                      240. 搜索二维矩阵||

                      参考题解&#xff1a;https://leetcode.cn/problems/search-a-2d-matrix-ii/solutions/2361487/240-sou-suo-er-wei-ju-zhen-iitan-xin-qin-7mtf 将矩阵旋转45度&#xff0c;可以看作一个二叉搜索树。 假设以左下角元素为根结点&#xff0c; 当target比root大的时候&#xff…

                      反向代理模块b

                      1 概念 1.1 反向代理概念 反向代理是指以代理服务器来接收客户端的请求&#xff0c;然后将请求转发给内部网络上的服务器&#xff0c;将从服务器上得到的结果返回给客户端&#xff0c;此时代理服务器对外表现为一个反向代理服务器。 对于客户端来说&#xff0c;反向代理就相当于…

                      【Linux权限】—— 于虚拟殿堂,轻拨密钥启华章

                      欢迎来到ZyyOvO的博客✨&#xff0c;一个关于探索技术的角落&#xff0c;记录学习的点滴&#x1f4d6;&#xff0c;分享实用的技巧&#x1f6e0;️&#xff0c;偶尔还有一些奇思妙想&#x1f4a1; 本文由ZyyOvO原创✍️&#xff0c;感谢支持❤️&#xff01;请尊重原创&#x1…

                      EasyExcel使用详解

                      文章目录 EasyExcel使用详解一、引言二、环境准备与基础配置1、添加依赖2、定义实体类 三、Excel 读取详解1、基础读取2、自定义监听器3、多 Sheet 处理 四、Excel 写入详解1、基础写入2、动态列与复杂表头3、样式与模板填充 五、总结 EasyExcel使用详解 一、引言 EasyExcel 是…

                      前端-Rollup

                      Rollup 是一个用于 JavaScript 的模块打包工具&#xff0c;它将小的代码片段编译成更大、更复杂的代码&#xff0c;例如库或应用程序。它使用 JavaScript 的 ES6 版本中包含的新标准化代码模块格式&#xff0c;而不是以前的 CommonJS 和 AMD 等特殊解决方案。ES 模块允许你自由…

                      vue3相关知识点

                      title: vue_1 date: 2025-01-28 12:00:00 tags:- 前端 categories:- 前端vue3 Webpack ~ vite vue3是基于vite创建的 vite 更快一点 一些准备工作 准备后如图所示 插件 Main.ts // 引入createApp用于创建应用 import {createApp} from vue // 引入App根组件 import App f…

                      微服务网关鉴权之sa-token

                      目录 前言 项目描述 使用技术 项目结构 要点 实现 前期准备 依赖准备 统一依赖版本 模块依赖 配置文件准备 登录准备 网关配置token解析拦截器 网关集成sa-token 配置sa-token接口鉴权 配置satoken权限、角色获取 通用模块配置用户拦截器 api模块配置feign…

                      华为小米vivo向上,苹果荣耀OPPO向下

                      日前&#xff0c;Counterpoint发布的手机销量月度报告显示&#xff0c;中国智能手机销量在2024年第四季度同比下降3.2%&#xff0c;成为2024年唯一出现同比下滑的季度。而对于各大智能手机品牌来说&#xff0c;他们的市场份额和格局也在悄然发生变化。 华为逆势向上 在2024年第…