登录
首页 >  文章 >  java教程

Java静态代码块与实例代码块区别详解

时间:2026-02-12 17:34:32 124浏览 收藏

大家好,今天本人给大家带来文章《Java静态代码块与实例代码块区别解析》,文中内容主要涉及到,如果你对文章方面的知识点感兴趣,那就请各位朋友继续看下去吧~希望能真正帮到你们,谢谢!

静态代码块只在类加载时执行一次,用于类级别初始化;实例代码块每次new对象时执行,用于对象初始化;二者执行顺序固定且不可混淆。

Java静态代码块与实例代码块的使用场景

静态代码块只在类加载时执行一次

静态代码块 static { ... } 在类第一次被 JVM 加载并初始化时运行,且仅执行一次。它不依赖任何对象实例,适合做类级别的资源准备,比如初始化静态配置、注册驱动、预热缓存。

常见错误是误以为每次 new 对象都会触发静态块——实际不会。如果你看到日志里静态块反复打印,大概率是类被多个 ClassLoader 重复加载(如 Web 应用中热部署、OSGi、模块化环境)。

  • 不能访问非静态成员变量或方法(编译报错:non-static variable xxx cannot be referenced from a static context
  • 执行顺序早于任何构造方法,也早于实例代码块
  • 多个静态块按源码顺序依次执行,可用来分段初始化不同模块

实例代码块在每次 new 对象时都执行

实例代码块 { ... }(无 static 修饰)会在每次调用构造方法前执行,且在构造方法体之前。它的作用相当于把重复的初始化逻辑从多个构造方法中抽出来,避免复制粘贴。

典型使用场景是:类有多个重载构造器,但都有共同的字段赋值、状态校验或资源预分配逻辑。

  • 可以访问静态和非静态成员(包括 this
  • 执行时机在构造方法的隐式或显式 super() 调用之后、构造方法体之前
  • 多个实例块也按源码顺序执行;若与字段初始化混合,注意字段声明位置会影响实际赋值顺序

静态块 vs 实例块:别在静态块里 new 实例

虽然语法允许在静态块中写 new MyClass(),但这会强制触发该类的初始化流程,可能引发意外的递归或死锁,尤其当构造过程又间接依赖当前类的其他静态资源时。

更隐蔽的问题是:如果实例初始化抛出异常(比如 NullPointerException),整个类加载失败,后续所有对该类的引用都会抛 NoClassDefFoundError,堆栈里却看不到原始异常原因。

  • 静态块中应只做轻量、确定性高的操作(如赋值常量、注册单例、设置系统属性)
  • 需要“首次使用才创建实例”的逻辑,优先用懒汉式单例(配合 volatile + 双重检查)或 java.util.concurrent.ConcurrentHashMap 缓存
  • 若必须在类加载阶段触发某些副作用(如启动后台线程),确保线程不阻塞、不依赖本类未初始化完成的部分

看一眼执行顺序就清楚了

下面这段代码能直观反映三者的执行时序:

public class InitOrder {
    static { System.out.println("1. 静态块"); }
    { System.out.println("2. 实例块"); }
    public InitOrder() { System.out.println("3. 构造方法"); }
    public static void main(String[] args) {
        System.out.println("开始 new");
        new InitOrder();
        new InitOrder();
    }
}

输出是:

1. 静态块
开始 new
2. 实例块
3. 构造方法
2. 实例块
3. 构造方法

注意:静态块只在 main 方法第一次触及该类时触发,不是在 javac 编译时,也不是在 main 方法声明处。

真正容易被忽略的是「类加载」和「类初始化」的区别:同一个类被不同 ClassLoader 加载,会得到不同的类对象,各自拥有独立的静态变量和静态块执行记录——这在插件化、热更新、测试 mock 场景下特别关键。

今天关于《Java静态代码块与实例代码块区别详解》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!

前往漫画官网入口并下载 ➜
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>