登录
首页 >  文章 >  java教程

Java类加载顺序全解析

时间:2026-02-19 19:24:45 371浏览 收藏

本文深入剖析了Java类加载与初始化的核心机制,重点揭示了new Child()时为何先执行Grandparent静态块——这并非代码执行顺序的巧合,而是JVM严格遵循“父类优先原则”和类初始化五阶段(加载→验证→准备→解析→初始化)所决定的底层逻辑;文章厘清了静态字段与静态块按源码文本顺序在初始化阶段赋值、static final常量因编译期内联而绕过初始化、以及仅六种“主动使用”场景才会真正触发类初始化等关键认知误区,帮你穿透表象,直击JVM规范本质,彻底摆脱对类加载过程的经验式误解。

在Java里什么是类加载顺序_Java类加载流程解析

类加载顺序不是“代码执行顺序”,而是JVM对类结构进行初始化的严格阶段链:加载→验证→准备→解析→初始化,其中静态内容初始化(static字段和块)按源码声明顺序触发,且父类优先于子类。

为什么 new Child() 会先打印 Grandparent 静态块?

因为 JVM 触发类初始化时,必须确保其直接父类已初始化完毕——这是“父类优先原则”,由规范强制保证,与你是否显式调用无关。

  • 只要 Child 类首次被主动使用(如 new Child()、访问 Child.staticField),JVM 就检查它是否已初始化;未初始化则先触发 Parent 初始化,而 Parent 又强制先完成 Grandparent 初始化
  • 静态变量赋值和静态代码块按 Java 源码中从上到下的**文本顺序**执行,不是按声明类型(字段/块)分组
  • 准备阶段只设默认值(如 int 设为 0),真正赋值发生在 初始化 阶段——所以 static int x = 5;5 是在初始化时才写入

哪些操作会真正触发类初始化?

只有六种“主动使用”场景才会强制进入初始化阶段,其他如单纯声明引用、子类引用父类静态字段(非 final)、数组创建等,都不会触发子类初始化。

  • 遇到 newgetstaticputstaticinvokestatic 字节码指令,且对应类未初始化
  • 使用 java.lang.reflect 对类进行反射调用(如 Class.forName("X"),注意带参版本默认 initialize=true
  • 初始化子类时,父类尚未初始化
  • 虚拟机启动时标明的主类(即含 main 方法的类)
  • 使用 JDK 7+ 的动态语言支持(如 invokedynamic)首次解析 MethodHandle 且对应类未初始化
  • java.lang.invoke.MethodHandle 解析结果为 REF_getStatic/REF_putStatic/REF_invokeStatic,且对应类未初始化

为什么 static final 常量不触发类初始化?

因为编译期就内联了——JVM 在 准备 阶段直接把常量值塞进调用方的常量池,根本不会走到 初始化 阶段去读原类。

  • 前提是:字段是基本类型或 String,且用字面量或编译期可确定的表达式初始化,如 static final int PORT = 8080;
  • 若写成 static final int PORT = Integer.valueOf(8080);static final String NAME = System.getProperty("name");,则仍会触发初始化
  • 这个优化让常量访问零开销,但也意味着:如果通过反射修改该类的静态 final 字段(需先 setAccessible(true)),实际生效的是运行时值,但调用方看到的仍是编译期内联的老值

最容易被忽略的一点是:类加载的“五阶段”只是逻辑划分,真实执行中各阶段高度交叉——比如 解析 可能延迟到首次方法调用时才做(支持多态),而 验证 可能在加载过程中就部分执行。别死记流程图,要盯住“什么时机真正改变运行状态”。

本篇关于《Java类加载顺序全解析》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于文章的相关知识,请关注golang学习网公众号!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>