当前位置: 首页 > news >正文

java 类的实例化过程,其中的相关顺序 包括有继承的子类等复杂情况,静态成员变量的初始化顺序,这其中jvm在干什么

Java类的实例化过程及初始化顺序

Java类的实例化过程涉及多个步骤,特别是在存在继承关系和静态成员的情况下。下面我将详细解释整个过程,包括JVM在其中的角色。

1. 类加载阶段(JVM的工作)

在实例化一个类之前,JVM首先需要加载这个类:

  1. 加载:JVM查找并加载类的二进制数据(.class文件)

  2. 验证:确保加载的类正确无误,符合JVM规范

  3. 准备:为类的静态变量分配内存,并设置默认初始值(0, null, false等)

  4. 解析:将符号引用转换为直接引用

  5. 初始化:执行类的静态初始化代码和静态变量初始化

2. 实例化过程顺序(包含继承)

当使用new关键字创建对象时,完整的初始化顺序如下:

父类静态成员和静态代码块

  1. 父类的静态变量初始化

  2. 父类的静态代码块(按代码中的顺序执行)

子类静态成员和静态代码块

  1. 子类的静态变量初始化

  2. 子类的静态代码块(按代码中的顺序执行)

父类实例成员和构造代码块

  1. 父类的实例变量初始化

  2. 父类的构造代码块(按代码中的顺序执行)

  3. 父类的构造函数(如果没有显式调用其他构造函数,则调用默认无参构造)

子类实例成员和构造代码块

  1. 子类的实例变量初始化

  2. 子类的构造代码块(按代码中的顺序执行)

  3. 子类的构造函数

3. JVM在实例化过程中的工作

在整个实例化过程中,JVM负责:

  1. 内存分配:为对象和它的实例变量分配堆内存

  2. 初始化默认值:将所有实例变量设置为默认值(0, null, false等)

  3. 方法表设置:为对象设置方法表(用于动态绑定和多态)

  4. 执行初始化代码:按照上述顺序执行各种初始化代码

  5. 构造器链调用:确保父类构造器在子类构造器之前被调用

  6. 处理多态:在运行时确定实际要调用的方法版本

4. 示例代码

class Parent {static {System.out.println("Parent static block");}{System.out.println("Parent instance block");}Parent() {System.out.println("Parent constructor");}
}class Child extends Parent {static {System.out.println("Child static block");}{System.out.println("Child instance block");}Child() {System.out.println("Child constructor");}
}public class Main {public static void main(String[] args) {new Child();}
}

输出顺序:

Parent static block
Child static block
Parent instance block
Parent constructor
Child instance block
Child constructor

5. 特殊情况

  • 如果类中有多个静态变量和静态代码块,它们按照在代码中出现的顺序执行

  • 实例变量和构造代码块也是按照代码中的顺序执行

  • 如果构造函数中显式调用了其他构造函数(this()或super()),则先执行被调用的构造函数

  • 如果父类没有无参构造器,子类必须显式调用父类的某个构造器(super(...))

理解这些初始化顺序对于避免NullPointerException和其他初始化相关的问题非常重要,特别是在复杂的继承层次结构中。

关于 Java 类的实例化过程

✅ 整体顺序正确,但可以补充一些关键细节:

需要修正/补充的点:

  1. 静态初始化(JVM 类加载阶段)

    • 静态变量和静态代码块的初始化顺序 仅发生在类首次被主动使用时(比如 new、访问静态成员、Class.forName() 等),而不是在程序启动时全部加载。

    • 如果类已经被加载过(比如之前已经调用过静态方法),则不会再次执行静态初始化。

  2. 实例初始化

    • final 实例变量 的初始化必须在构造函数完成之前完成,否则编译错误。

    • 如果类没有显式构造函数,JVM 会提供默认无参构造,但如果父类没有无参构造,子类必须显式调用 super(...),否则编译失败。

  3. JVM 的隐含操作

    • 在构造函数真正执行前,JVM 会 隐式初始化所有实例变量为默认值0/null/false),然后才执行显式赋值或初始化块。

    • 如果存在 实例变量直接赋值(如 int x = 5;),这些赋值会在构造代码块之前执行。

示例修正:

class Parent {static { System.out.println("Parent static block"); }{ System.out.println("Parent instance block"); }Parent() { System.out.println("Parent constructor"); }
}class Child extends Parent {static { System.out.println("Child static block"); }{ System.out.println("Child instance block"); }int x = initX(); // 这个会在实例块之后执行吗?❌ 实际在实例块之前!Child() { System.out.println("Child constructor"); }private int initX() { System.out.println("Initializing x"); return 1; }
}public class Main {public static void main(String[] args) {new Child();}
}

实际输出:

Parent static block
Child static block
Parent instance block
Parent constructor
Initializing x  // 变量初始化在实例块之前!
Child instance block
Child constructor

1. final 实例变量必须在构造函数完成前初始化(否则编译错误)

final 实例变量必须在 对象构造完成前 被赋值,且只能赋值一次。有三种方式初始化:

  1. 声明时直接赋值

  2. 在实例初始化块中赋值

  3. 在每个构造函数中赋值

class FinalExample {final int x;          // ❌ 如果不初始化,编译报错final int y = 10;     // ✅ 方式1:声明时赋值final int z;{ z = 20; }           // ✅ 方式2:实例初始化块赋值FinalExample() {x = 30;           // ✅ 方式3:构造函数中赋值}FinalExample(int val) {x = val;         // ✅ 另一个构造函数也必须赋值}
}
错误示例:
class FinalError {final int x;  // ❌ 编译错误:未初始化final变量xFinalError() {// 忘记给x赋值}
}

2. 父类没有无参构造时,子类必须显式调用 super(...)

如果父类 没有无参构造,子类 必须 在构造函数的第一行显式调用 super(...),否则编译失败。

示例代码:
class Parent {Parent(int value) {  // ❗父类只有带参构造,没有默认无参构造System.out.println("Parent constructor: " + value);}
}class Child extends Parent {Child() {super(10);  // ✅ 必须显式调用父类构造,否则编译错误System.out.println("Child constructor");}
}public class Main {public static void main(String[] args) {new Child();}
}
输出:
Parent constructor: 10
Child constructor
错误示例:
class Child extends Parent {Child() {  // ❌ 编译错误:父类Parent中没有默认构造函数System.out.println("Child constructor");}
}

关键总结

  1. final 实例变量

    • 必须且只能赋值一次。

    • 必须在构造完成前初始化(三种方式选其一)。

  2. 构造函数继承规则

    • 如果父类 没有无参构造,子类 必须显式调用 super(...)

    • 如果父类有无参构造,子类可以不写 super()(JVM 会隐式调用)。

这些规则由 Java 语言规范(JLS) 强制约束,编译器会严格检查。

    http://www.xdnf.cn/news/172603.html

    相关文章:

  • xe-upload上传文件插件
  • WPF常用技巧汇总 - Part 2
  • Qt项目全局设置UTF-8编码方法(MSVS编译中文报错解决办法)
  • 新能源汽车运动控制器核心芯片选型与优化:MCU、DCDC与CANFD协同设计
  • 设计一个新能源汽车控制系统开发框架,并提供一个符合ISO 26262标准的模块化设计方案。
  • Java高频常用工具包汇总
  • [特殊字符]实战:使用 Canal + MQ + ES + Redis + XXL-Job 打造高性能地理抢单系统
  • Spark Mllib 机器学习
  • 第二章,网络类型及数据链路层协议
  • SMART:大模型在关键推理步骤辅导小模型,在保持高推理效率的同时,显著提升小模型的推理能力!!
  • python合并一个word段落中的run
  • 决策树相关案例
  • 【Node.js 】在Windows 下搭建适配 DPlayer 的轻量(简陋)级弹幕后端服务
  • Linux系统之设置开机启动运行桌面环境
  • 力扣hot100_子串_python版本
  • Nginx配置文件介绍
  • 机器学习day2-seaborn绘图练习
  • 数模学习:二,MATLAB的基本语法使用
  • 跨专业自学AI人工智能学习路线图(2025版)
  • Android完整开发环境搭建/Studio安装/NDK/本地Gradle下载配置/创建AVD/运行一个Android项目/常用插件
  • 金融数据分析(Python)个人学习笔记(13):自然语言处理
  • Kubernetes学习笔记-配置Service对接第三方访问
  • 【Redis】服务端高并发分布式结构演进之路
  • 零基础小白如何上岸数模国奖
  • IDEA 连接 Oracle 数据库
  • 安卓7.0以上抓包配置--Charles
  • ​​全栈自动化:从零构建智能CI/CD流水线​
  • 手搓传染病模型(SEIR)
  • k8s的volume
  • Alibaba Cloud Linux 3.2104 LTS 64位 容器优化版安装docker docker compose记录