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

静态代码块只在类加载时执行一次
静态代码块 static { ... } 在类第一次被 JVM 加载并初始化时运行,且仅执行一次。它不依赖任何对象实例,适合做类级别的资源准备,比如初始化静态配置、注册驱动、预热缓存。
常见错误是误以为每次 new 对象都会触发静态块——实际不会。如果你看到日志里静态块反复打印,大概率是类被多个 ClassLoader 重复加载(如 Web 应用中热部署、OSGi、模块化环境)。
- 不能访问非静态成员变量或方法(编译报错:
non-static variable xxx cannot be referenced from a static context) - 执行顺序早于任何构造方法,也早于实例代码块
- 多个静态块按源码顺序依次执行,可用来分段初始化不同模块
实例代码块在每次 new 对象时都执行
实例代码块 { ... }(无 static 修饰)会在每次调用构造方法前执行,且在构造方法体之前。它的作用相当于把重复的初始化逻辑从多个构造方法中抽出来,避免复制粘贴。
典型使用场景是:类有多个重载构造器,但都有共同的字段赋值、状态校验或资源预分配逻辑。
立即学习“Java免费学习笔记(深入)”;
- 可以访问静态和非静态成员(包括
this) - 执行时机在构造方法的隐式或显式
super()调用之后、构造方法体之前 - 多个实例块也按源码顺序执行;若与字段初始化混合,注意字段声明位置会影响实际赋值顺序
静态块 vs 实例块:别在静态块里 new 实例
虽然语法允许在静态块中写 new MyClass(),但这会强制触发该类的初始化流程,可能引发意外的递归或死锁,尤其当构造过程又间接依赖当前类的其他静态资源时。
使用模板与程序分离的方式构建,依靠专门设计的数据库操作类实现数据库存取,具有专有错误处理模块,通过 Email 实时报告数据库错误,除具有满足购物需要的全部功能外,成新商城购物系统还对购物系统体系做了丰富的扩展,全新设计的搜索功能,自定义成新商城购物系统代码功能代码已经全面优化,杜绝SQL注入漏洞前台测试用户名:admin密码:admin888后台管理员名:admin密码:admin888
更隐蔽的问题是:如果实例初始化抛出异常(比如 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 场景下特别关键。









