完整的类在JVM中的生命周期详解

news2025/12/16 15:25:52

首先给出一个示例代码:

        示例的目标是展示一个多功能的类结构,包含继承、接口实现、静态成员、本地方法、线程安全等特性,同时模拟一个简单的“计算器”场景,计算并管理数字。(尽量将所有的 Java 组件和关键字都给出,完整给出全面的生命周期)


示例代码及其作用

        这个示例实现了一个计算器类 Calculator,它继承自一个抽象父类 BaseCalculator,并实现了一个接口 Operable。它包含:

  • 静态变量和方法来跟踪计算次数。
  • 本地方法(假设由 C/C++ 实现)来执行特殊计算。
  • 同步方法确保多线程安全。
  • 嵌套类和枚举用于扩展功能。
  • 子类 AdvancedCalculator 进一步扩展功能。

以下是带详细注释的完整代码:

package com.example;

// 抽象父类,提供基础计算功能
abstract class BaseCalculator {
    // 受保护的实例变量,表示基础值
    protected int baseValue = 10;

    // 抽象方法,要求子类实现具体的计算逻辑
    abstract int calculateBase(int input);

    // 普通方法,显示基础值
    void displayBase() {
        System.out.println("基础值: " + baseValue);
    }
}

// 接口,定义可操作的计算行为
interface Operable {
    // 接口方法,执行加法操作
    int add(int a, int b);
}

// 主类:计算器类,继承 BaseCalculator 并实现 Operable 接口
public class Calculator extends BaseCalculator implements Operable {
    // 静态常量,记录最大允许的计算次数
    public static final int MAX_CALCULATIONS = 100;

    // 静态变量,跟踪全局计算次数
    private static int calculationCount = 0;

    // 实例变量,表示当前计算结果,volatile 确保线程可见性
    private volatile int currentResult = 0;

    // transient 变量,不会序列化,用于临时存储数据
    private transient String tempLog = "Calculation started";

    // 静态初始化块,在类加载时执行一次
    static {
        System.out.println("已加载Calculator类。最大允许的计算次数: " + MAX_CALCULATIONS);
    }

    // 实例初始化块,每次创建对象时执行
    {
        System.out.println("新建Calculator实例。");
        calculationCount++; // 增加计算次数
    }

    // 构造方法,初始化当前结果
    public Calculator(int initialValue) {
        super(); // 调用父类构造方法
        this.currentResult = initialValue;
    }

    // 静态方法,返回当前的计算次数
    public static int getCalculationCount() {
        return calculationCount;
    }

    // 本地方法,假设由 C/C++ 实现,用于特殊计算(这里仅声明)
    public native int specialCalculation(int input);

    // 重写抽象方法,计算基于 baseValue 的结果
    @Override
    int calculateBase(int input) {
        return baseValue + input;
    }

    // 实现接口方法,执行加法操作
    @Override
    public int add(int a, int b) {
        currentResult = a + b;
        return currentResult;
    }

    // 同步方法,确保多线程环境下安全更新结果
    public synchronized void multiply(int factor) {
        currentResult *= factor;
        System.out.println("乘法结果: " + currentResult);
    }

    // 获取当前结果的方法
    public int getCurrentResult() {
        return currentResult;
    }

    // 嵌套静态类,用于日志记录
    static class CalculationLogger {
        // 日志记录方法
        void log(String message) {
            System.out.println("Log: " + message);
        }
    }

    // 嵌套枚举,表示计算类型
    enum OperationType {
        ADD, MULTIPLY, SPECIAL
    }

    // 主方法,测试计算器功能
    public static void main(String[] args) {
        // 创建计算器实例
        Calculator calc = new Calculator(5);
        
        // 测试继承的父类方法
        calc.displayBase();
        
        // 测试抽象方法实现
        System.out.println("基础计算结果: " + calc.calculateBase(20));
        
        // 测试接口方法
        System.out.println("加法结果: " + calc.add(10, 15));
        
        // 测试同步方法
        calc.multiply(2);
        
        // 测试静态方法
        System.out.println("总计算次数: " + Calculator.getCalculationCount());
        
        // 测试嵌套类
        CalculationLogger logger = new CalculationLogger();
        logger.log("Calculation completed");
        
        // 测试枚举
        OperationType op = OperationType.ADD;
        System.out.println("操作类型: " + op);
        
        // 检查实例类型
        if (calc instanceof Calculator) {
            System.out.println("实例是计算器");
        }
    }
}

// 子类:高级计算器,扩展 Calculator 的功能
class AdvancedCalculator extends Calculator {
    // 子类构造方法
    public AdvancedCalculator(int initialValue) {
        super(initialValue);
    }

    // 子类新增方法,计算平方
    public int square() {
        int result = getCurrentResult() * getCurrentResult();
        System.out.println("平方的结果: " + result);
        return result;
    }
}


代码作用和功能说明

  1. 类结构和继承

    • BaseCalculator 是一个抽象父类,提供基础值和方法,模拟计算器的基本功能。
    • Calculator 继承 BaseCalculator,并实现 Operable 接口,提供加法等操作。
    • AdvancedCalculator 是 Calculator 的子类,增加了平方计算功能。
  2. 静态成员

    • MAX_CALCULATIONS 是静态常量,限制计算次数(示例中未强制执行,仅展示概念)。
    • calculationCount 是静态变量,记录创建的计算器实例数。
    • getCalculationCount() 是静态方法,提供全局访问。
  3. 本地方法

    • specialCalculation() 使用 native 关键字,假设由本地代码实现特殊计算(实际需要 JNI 实现)。
  4. 线程安全

    • multiply() 使用 synchronized 确保多线程环境下安全更新 currentResult
    • volatile 修饰 currentResult,保证线程间可见性。
  5. 嵌套类和枚举

    • CalculationLogger 是静态嵌套类,用于记录计算日志。
    • OperationType 是枚举,定义支持的操作类型。
  6. 功能实现

    • 支持加法(add)、乘法(multiply)、平方(square)等计算。
    • 提供结果查询(getCurrentResult)和计算次数统计(getCalculationCount)。
    • 通过 main 方法测试所有功能。

输出示例

运行 main 方法的输出(specialCalculation 未实现):

已加载Calculator类。最大允许的计算次数:100
新建Calculator实例。
基础值:10
基础计算结果:30
加法结果:25
乘法结果:50
总计算次数:1
Log: Calculation completed
操作类型: ADD
实例是计算器


        下面详细分析 Java 8 的环境下,上面 Calculator 类从创建到被垃圾回收(GC)销毁的完整流程,涵盖 PC 寄存器、Java 虚拟机栈、Java 堆、方法区、运行时常量池和本地方法栈六个关键部分。我会结合 JVM 的运行时数据区,逐步说明每个阶段的内存分配和回收过程,并尽量贴近底层实现(基于 HotSpot JVM 的典型行为)。


1. 类的加载与初始化(方法区和运行时常量池)

流程概述

        在 Calculator 类首次使用(如 Calculator calc = new Calculator(5);)时,JVM 会通过类加载器加载该类及其相关类(BaseCalculatorOperableAdvancedCalculator 等)。这涉及到方法区和运行时常量池的初始化。

底层原理
  1. 类加载

    • JVM 使用类加载器(如 AppClassLoader)加载 Calculator.class 文件。
    • 类文件被解析后,JVM 在方法区(Java 8 中为 Metaspace)分配内存,存储类的元数据,包括:
      • 类名(com.example.Calculator
      • 父类(BaseCalculator
      • 接口(Operable
      • 字段信息(MAX_CALCULATIONScalculationCount 等)
      • 方法表(addmultiplyspecialCalculation 等)
      • 字节码(每个方法的指令序列)。
  2. 运行时常量池

    • 方法区中的一部分逻辑区域为运行时常量池(Runtime Constant Pool),存储类文件中 constant_pool 表的运行时表示。
    • 例如,MAX_CALCULATIONS = 100 的值 100、字符串 "已加载Calculator类..."、方法引用(如 System.out.println)等都被放入常量池。
    • 在 HotSpot JVM 中,运行时常量池的实现依赖于 Metaspace 中的 ConstantPool 对象。
  3. 静态初始化

    • 类加载完成后,JVM 执行 <clinit> 方法(类初始化方法),运行静态初始化块:
      static {
          System.out.println("已加载Calculator类。最大允许的计算次数: " + MAX_CALCULATIONS);
      }
      
    • 这会在方法区中为 calculationCount 分配内存并初始化为 0,并在 Java 堆中分配字符串对象(常量池引用)。
源代码层面
  • HotSpot JVM 的 InstanceKlass 类表示方法区中的类元数据。
  • ConstantPool 对象管理运行时常量池,存储符号引用(如字段和方法的 CONSTANT_Fieldref_info)。

2. 对象创建(Java 堆和 Java 虚拟机栈)

流程概述

        当执行 Calculator calc = new Calculator(5);  时,JVM 创建 Calculator 对象实例,并分配栈帧来执行构造方法。

底层原理
  1. Java 堆分配

    • JVM 在堆的年轻代(Young Generation)中的 Eden 区为 Calculator 对象分配内存。
    • 对象包含实例字段:
      • baseValue(继承自 BaseCalculator
      • currentResult
      • tempLog
    • 对象头(Object Header)包含元数据,如指向方法区中 Calculator 类元信息的指针(klass 指针)。
  2. Java 虚拟机栈

    • 为调用 Calculator(int) 构造方法,JVM 在当前线程的栈中推送一个新的栈帧(Frame)。
    • 栈帧结构:
      • 局部变量表:存储 this(对象引用)和参数 initialValue
      • 操作数栈:用于计算(如 this.currentResult = initialValue)。
      • 常量池引用:指向运行时常量池中构造方法的相关条目。
    • 执行流程:
      • super(); 调用 BaseCalculator 的构造方法,分配一个父类栈帧。
      • this.currentResult = initialValue; 将 5 存入堆中的对象字段。
      • 实例初始化块 {} 执行,更新 calculationCount
  3. PC 寄存器

    • PC(Program Counter)寄存器记录当前线程正在执行的字节码指令地址。
    • 例如,在构造方法中,PC 指向 invokespecial(调用 super())或 putfield(设置 currentResult)指令。
源代码层面
  • HotSpot 的 oopDesc 表示堆中的对象,markOop 是对象头。
  • frame 类管理栈帧,interpreter 模块执行字节码。

3. 方法调用(Java 虚拟机栈和本地方法栈)

流程概述

        执行 calc.add(10, 15); 和 calc.specialCalculation(20); 时,JVM 分别调用普通方法和本地方法。

底层原理
  1. 普通方法调用(Java 虚拟机栈)

    • 为 add(int, int) 创建栈帧:
      • 局部变量表:thisa(10)、b(15)。
      • 操作数栈:计算 a + b,结果存入 currentResult
    • PC 寄存器更新为 add 方法的字节码地址。
    • 方法返回后,栈帧弹出。
  2. 本地方法调用(本地方法栈)

    • 调用 specialCalculation(int) 时,JVM 使用 Java Native Interface(JNI)。
    • 在本地方法栈(Native Method Stack)分配栈帧,存储 JNI 调用参数。
    • PC 寄存器值未定义(本地方法不由 JVM 解释执行)。
    • JNI 调用本地库(如 .dll 或 .so 文件),假设返回结果。
  3. 同步方法(Java 虚拟机栈)

    • 调用 multiply(2) 时,栈帧包含同步锁信息。
    • JVM 使用对象头的监视器(Monitor)实现 synchronized,确保线程安全。
源代码层面
  • InterpreterRuntime::resolve_invoke 解析方法调用。
  • jni.h 和 JVM_ENTRY 宏处理本地方法栈。

4. 对象使用(Java 堆和方法区交互)

流程概述

        对象 calc 被使用时,JVM 通过堆和方法区的协作执行逻辑。

底层原理
  • 堆中的对象
    • currentResult 更新为 25(add)、50(multiply)。
    • tempLog 指向堆中的字符串对象(若非常量池引用)。
  • 方法区
    • 方法字节码从方法区加载到栈帧执行。
    • 嵌套类 CalculationLogger 和枚举 OperationType 的元数据也在方法区。

5. 垃圾回收(GC)过程(Java 堆)

流程概述

        当 calc 不再被引用(如 calc = null; 或离开作用域),JVM 的垃圾回收器(GC)回收其内存。

底层原理
  1. 可达性分析

    • GC 从根集(GC Roots)开始标记:
      • 栈中的局部变量(如 calc)。
      • 方法区中的静态变量(如 calculationCount)。
    • 若 calc 无引用,则标记为不可达。
  2. 年轻代回收(Minor GC)

    • calc 在 Eden 区创建,若仍存活,可能晋升到 Survivor 区。
    • Minor GC 使用复制算法(Copying Algorithm):
      • 将 Survivor 中的存活对象移动到另一个 Survivor 或老年代。
      • 清空 Eden 和当前 Survivor。
  3. 老年代回收(Major GC)

    • 若 calc 存活多次 Minor GC,晋升到老年代(Tenured Generation)。
    • 使用标记-清除(Mark-Sweep)或标记-整理(Mark-Compact)算法:
      • 标记存活对象。
      • 清除不可达对象(如 calc),回收内存。
  4. Metaspace(方法区)回收

    • 若 Calculator 类不再被使用(无实例且类加载器可回收),Metaspace 中的元数据和常量池条目被回收。
    • Java 8 中 Metaspace 使用本地内存,受 -XX:MaxMetaspaceSize 控制。
源代码层面
  • GenCollectedHeap 管理堆的分代。
  • G1CollectedHeap(默认 GC)实现分代回收。
  • MetaspaceUtils 处理 Metaspace 回收。

6. 完整流程总结

  1. 方法区和常量池:类加载,元数据和常量存储。
  2. Java 堆:对象在 Eden 区创建,字段存储实例数据。
  3. Java 虚拟机栈:栈帧执行构造方法和普通方法。
  4. PC 寄存器:跟踪指令地址。
  5. 本地方法栈:执行 native 方法。
  6. GC 销毁:对象不可达后,Minor/Major GC 回收堆内存,Metaspace 回收类元数据。

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

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

相关文章

C#里使用libxl的数字格式

由于EXCEL里可以表示不同的数字格式, 比如表示货币数字时,与表示普通序号的数字就不一样。 还有科学计算表示的数字使用小数点位数与普通货币也不一样。 如下所示: 要使用这些格式, 下面创建一个例子来演示保存这些数字格式: private void button11_Click(object send…

c#难点整理2

1.对象池的使用 就是先定义一系列的对象&#xff0c;用一个&#xff0c;调一个。 public class ObjectPool<T> where T : new(){private Queue<T> pool; // 用于存储对象的队列private int maxSize; // 对象池的最大容量// 构造函数public ObjectPool(int maxSi…

解锁物联网高效开发,Synaptics SYN43756E Wi-Fi 6E 芯片登场

Synaptics 的 SYN43756E 芯片是一款高性能的 Wi-Fi 6E 支持 11a/b/g/n/ac/ax 的物联网&#xff08;IoT&#xff09;SoC&#xff0c;具备多项先进特性&#xff0c;适用于多种应用场景&#xff0c;以下是其主要优势&#xff1a; 1. 广泛的应用场景 智慧家庭&#xff1a;支持多种…

DeepSeek 助力 Vue3 开发:打造丝滑的表格(Table)之添加导出数据功能

前言:哈喽,大家好,今天给大家分享一篇文章!并提供具体代码帮助大家深入理解,彻底掌握!创作不易,如果能帮助到大家或者给大家一些灵感和启发,欢迎收藏+关注哦 💕 目录 DeepSeek 助力 Vue3 开发:打造丝滑的表格(Table)之添加导出数据功能📚页面效果📚指令输入�…

2024年河南省职业院校 技能大赛高职组 “大数据分析与应用” 赛项任务书(四)

2024 年河南省职业院校 技能大赛高职组 “大数据分析与应用” 赛项任务书&#xff08;四&#xff09;&#xff09; 背景描述&#xff1a;任务一&#xff1a;Hadoop 完全分布式安装配置&#xff08;25 分&#xff09;任务二&#xff1a;离线数据处理&#xff08;25 分&#xff0…

dify创建第一个Agent

1、首先LLM模型必须支持 Function Calling 由于deepseek-R1本地化部署时还不支持&#xff0c;所以使用 qwq模型。 2、创建空白 Agent 3、为Agent添加工具 4、测试 当未添加时间工具时 询问 时间 如下 5、开启时间工具 询问如下

⭐算法OJ⭐判断二叉搜索树【树的遍历】(C++实现)Validate Binary Search Tree

图论入门【数据结构基础】&#xff1a;什么是树&#xff1f;如何表示树&#xff1f; 之前我们有分别讲解二叉树的三种遍历的相关代码实现&#xff1a; ⭐算法OJ⭐二叉树的前序遍历【树的遍历】&#xff08;C实现&#xff09;Binary Tree Preorder Traversal ⭐算法OJ⭐二叉树的…

2. 商城前端部署

商城客户端前端部署 https://gitee.com/newbee-ltd/newbee-mall-api-go 使用开源新蜂商城的前端&#xff0c;git clone到本地 然后在vscode终端依次输入下列指令&#xff08;配置好vue3相关环境的前提下&#xff09;&#xff1a; npm install npm i --legacy-peer-deps npm …

鸿蒙生态开发

鸿蒙生态开发概述 鸿蒙生态是华为基于开源鸿蒙&#xff08;OpenHarmony&#xff09;构建的分布式操作系统生态&#xff0c;旨在通过开放共享的模式连接智能终端设备、操作系统和应用服务&#xff0c;覆盖消费电子、工业物联网、智能家居等多个领域。以下从定义与架构、核心技术…

基于STM32进行FFT滤波并计算插值DA输出

文章目录 一、前言背景二、项目构思1. 确定FFT点数、采样率、采样点数2. 双缓存设计 三、代码实现1. STM32CubeMX配置和HAL库初始化2. 核心代码 四、效果展示和后话五、项目联想与扩展1. 倍频2. 降频3. 插值3.1 线性插值3.2 样条插值 一、前言背景 STM32 对 AD 采样信号进行快…

【Oracle资源损坏类故障】:详细了解坏块

目录 1、物理坏块与逻辑坏块 1.1、物理坏块 1.2、逻辑坏块 2、两个坏块相关的参数 2.1、db_block_checksum 2.2、db_block_checking 3、检测坏块 3.1、告警日志 3.2、RMAN 3.3、ANALYZE 3.4、数据字典 3.5、DBVERIFY 4、修复坏块 4.1、RMAN修复 4.2、DBMS_REPA…

996引擎-接口测试:背包

996引擎-接口测试:背包 背包测试NPC参考资料背包测试NPC CONSTANT = require("Envir/QuestDiary/constant/CONSTANT.lua"); MsgUtil = require("Envir/QuestDiary/utils/996/MsgUtil.lua");

Electron打包文件生成.exe文件打开即可使用

1 、Electron 打包&#xff0c;包括需要下载的内容和环境配置步骤 注意&#xff1a;Electron 是一个使用 JavaScript、HTML 和 CSS 构建跨平台桌面应用程序的框架 首先需要电脑环境有Node.js 和 npm我之前的文章有关nvm下载node的说明也可以去官网下载 检查是否有node和npm环…

单播、广播、组播和任播

文章目录 一、单播二、广播三、组播四、任播代码示例&#xff1a; 五、各种播的比较 一、单播 单播&#xff08;Unicast&#xff09;是一种网络通信方式&#xff0c;它指的是在网络中从一个源节点到一个单一目标节点对的传输模式。单播传输时&#xff0c;数据包从发送端直接发…

Cursor+Claude-3.5生成Android app

一、Android Studio下载 https://developer.android.com/studio?hlzh-tw#get-android-studio 等待安装完成 二、新建工程 点击new project 选择Empty Activity 起一个工程名 当弹出这个框时 可以在settings里面选择No proxy 新建好后如下 点击右边模拟器&#xff0c…

QT Quick(C++)跨平台应用程序项目实战教程 3 — 项目基本设置(窗体尺寸、中文标题、窗体图标、可执行程序图标)

目录 1. 修改程序界面尺寸和标题 2. 窗体图标 3. 修改可执行程序图标 上一章创建好了一个初始Qt Quick项目。本章介绍基本的项目修改方法。 1. 修改程序界面尺寸和标题 修改Main.qml文件&#xff0c;将程序宽度设置为1200&#xff0c;程序高度设置为800。同时修改程序标题…

Transformers x SwanLab:可视化NLP模型训练(2025最新版)

HuggingFace 的 Transformers 是目前最流行的深度学习训框架之一&#xff08;100k Star&#xff09;&#xff0c;现在主流的大语言模型&#xff08;LLaMa系列、Qwen系列、ChatGLM系列等&#xff09;、自然语言处理模型&#xff08;Bert系列&#xff09;等&#xff0c;都在使用T…

VSCode 抽风之 两个conda环境同时在被激活

出现了神奇的(toolsZCH)(base) 提示符&#xff0c;如下图所示&#xff1a; 原因大概是&#xff1a;conda 环境的双重激活&#xff1a;可能是 conda 环境没有被正确清理或初始化&#xff0c;导致 base 和 toolsZCH 同时被激活。 解决办法就是 &#xff1a;conda deactivate 两次…

Mybatis的基础操作——03

写mybatis代码的方法有两种&#xff1a; 注解xml方式 本篇就介绍XML的方式 使用XML来配置映射语句能够实现复杂的SQL功能&#xff0c;也就是将sql语句写到XML配置文件中。 目录 一、配置XML文件的路径&#xff0c;在resources/mapper 的目录下 二、写持久层代码 1.添加mappe…

React:React主流组件库对比

1、Material-UI | 官网 | GitHub | GitHub Star: 94.8k Material-UI 是一个实现了 Google Material Design 规范的 React 组件库。 Material UI 包含了大量预构建的 Material Design 组件&#xff0c;覆盖导航、滑块、下拉菜单等各种常用组件&#xff0c;并都提供了高度的可定制…