Java代码执行顺序解析与实例
时间:2026-02-21 17:32:38 477浏览 收藏
本文深入剖析了Java程序从启动到对象创建全过程中的真实执行顺序,揭示了看似简单的代码背后由JVM类加载机制和实例化协议严格规定的隐性时序:静态成员按文本顺序且父类优先初始化,实例创建则遵循“内存分配→父类构造→子类非静态块→子类构造器”的固定链条;特别强调static final编译期常量的内联特性、父类构造器中调用重写方法的风险、以及main方法远非执行起点等关键陷阱,并通过典型输出示例和字节码视角说明——真正决定代码何时运行的不是书写位置或缩进,而是JVM规范定义的加载、链接、初始化三阶段与对象生命周期协议,帮助开发者摆脱“所见即所得”的直觉误区,精准掌控程序行为。

类加载时静态成员的初始化顺序
Java程序启动时,并不是从 main 方法第一行开始“执行”的,而是先触发类加载和链接,再进行初始化——这个阶段会按源码中出现的**文本顺序**执行所有 static 变量赋值和 static 代码块。
容易忽略的是:父类的静态成员总在子类之前初始化,哪怕子类的 static 块写在文件顶部;而同一类中,static final 基本类型常量(如 static final int X = 1;)属于编译期常量,不参与运行时初始化流程,会被直接内联。
- 多个
static块之间严格按书写顺序执行 static变量初始化表达式中若调用方法,该方法内对尚未初始化的static变量读取,结果为默认值(如int是0)- 避免在
static初始化中依赖尚未定义的其他static成员,否则行为不可靠
实例创建时非静态成员的执行顺序
当执行 new MyClass() 时,JVM 按固定顺序处理:先分配内存(字段设默认值),再执行父类构造器链,最后执行当前类的非静态变量赋值和非静态代码块,最终进入当前类构造方法体。
这意味着:即使你在构造方法里把某个字段设为 5,它可能已经被前面的非静态块设成了 3;而父类构造器中若调用了被子类重写的方法,此时子类字段仍处于默认值状态(常见坑)。
- 父类构造器 → 子类字段默认值 → 子类非静态块 → 子类构造器参数赋值 → 子类构造器方法体
- 非静态块和非静态字段初始化语句,按它们在类中出现的文本顺序执行
- 不要在父类构造器中调用
final以外的可重写方法,子类字段尚未初始化
main 方法只是入口,不是执行起点
main 方法是 JVM 启动后查找并调用的第一个用户方法,但它所在类的静态初始化早已完成。如果 main 所在类有 static 块或静态字段初始化,这些代码会在 main 开始前就执行完毕。
典型反例:有人在 main 里打印日志,却发现在此之前已有输出——那一定是类的静态初始化块干的。
- JVM 启动 → 加载包含
main的类 → 执行其静态初始化 → 调用main - 如果
main类依赖其他类(比如声明了static Other o = new Other();),则那些类也会被递归触发加载和初始化 - 没有被显式引用的类,即使存在也不会被加载(懒加载原则)
字节码层面的执行控制流
Java 源码的执行顺序,在字节码中体现为 invokestatic、invokespecial、invokevirtual 等指令的排列,但真正决定“何时跑哪段”的是 JVM 的类加载机制和对象实例化协议,不是语法糖或缩进。
例如,Lambda 表达式会被编译成私有静态方法,其执行时机仍服从所在位置的上下文顺序;而 try-with-resources 中的资源关闭,由编译器插入到 finally 块,实际执行发生在对应作用域退出时,而非语句写在哪一行。
- 条件分支(
if)、循环(for)等控制结构,只影响运行时跳转,不改变类/实例初始化阶段的固有顺序 - 多线程环境下,不同线程看到的静态初始化完成状态可能不同,需靠
clinit锁保证单次执行,但普通字段读写无此保障 - 反射调用
Class.forName("X")会强制触发初始化;而ClassLoader.loadClass("X")不会
class Parent {
static { System.out.println("Parent static"); }
{ System.out.println("Parent init"); }
Parent() { System.out.println("Parent ctor"); }
}
class Child extends Parent {
static { System.out.println("Child static"); }
{ System.out.println("Child init"); }
Child() { System.out.println("Child ctor"); }
}
public class Main {
public static void main(String[] args) {
new Child();
}
}
上面代码输出顺序固定为:Parent static → Child static → Parent init → Parent ctor → Child init → Child ctor。这个顺序由 JVM 规范硬性规定,不是 Java 编译器“优化”出来的。
真正难调试的,往往不是语句本身怎么写,而是你没意识到某段逻辑其实在 main 之前或构造器中途就被执行了。
好了,本文到此结束,带大家了解了《Java代码执行顺序解析与实例》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多文章知识!
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
235 收藏
-
431 收藏
-
252 收藏
-
253 收藏
-
263 收藏
-
398 收藏
-
445 收藏
-
323 收藏
-
471 收藏
-
233 收藏
-
404 收藏
-
393 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习