news 2026/6/9 18:31:39

java中<clinit>()与<init>()区别

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
java中<clinit>()与<init>()区别

一、先明确两个 “构造方法” 的核心区别

Java 中有两种不同的 “构造方法”,二者的作用、执行时机完全无关:

构造方法类型名称(字节码层面)通俗理解手动定义方式核心作用
类构造方法<clinit>()静态构造器、类初始化方法无需手动定义(编译器自动生成)初始化类的静态资源(静态变量、静态代码块)
实例构造方法<init>()对象构造器、构造函数public 类名(...) { ... }初始化对象的实例资源(实例变量、实例代码块)

二、类加载初始化阶段:仅执行类构造方法<clinit>(),不执行实例构造方法<init>()

类加载的「初始化阶段」是类加载流程的第 5 步(加载→验证→准备→解析→初始化),其核心任务就是执行类构造方法<clinit>(),而实例构造方法<init>()与类加载无关,仅在创建对象(new关键字、反射等)时执行。

1. 类构造方法<clinit>()的核心特性
  • 自动生成:由编译器自动收集类中的「静态变量显式赋值语句」和「静态代码块(static{})」按书写顺序合并生成,无需开发者手动定义;
  • 仅执行一次:JVM 保证一个类的<clinit>()方法在多线程环境下被同步执行,且仅执行一次(类初始化的原子性);
  • 父类优先执行:若当前类有父类(且父类未初始化),JVM 会先执行父类的<clinit>()方法,再执行当前类的<clinit>()
  • 无参数无返回值:字节码层面的特殊方法,无法通过反射直接调用(可通过Class.forName()间接触发);
  • 可选生成:若类中没有静态变量显式赋值,也没有静态代码块,编译器不会生成<clinit>()方法,初始化阶段无需执行任何操作。
2. 实例构造方法<init>()的执行时机

实例构造方法<init>()是对象实例化阶段的方法,与类加载的初始化阶段无关,只有通过以下方式创建对象时才会执行:

  • new 类名():直接创建对象;
  • 反射:Class.newInstance()(已过时)、Constructor.newInstance()
  • 反序列化:从字节流恢复对象时;
  • 克隆:Object.clone()(需实现Cloneable接口)。

三、代码示例:验证初始化阶段的执行行为

示例 1:类初始化阶段执行<clinit>()(静态资源初始化)

java

运行

public class ClassInitDemo { // 1. 静态变量显式赋值(会被收集到<clinit>()中) private static int staticVar = initStaticVar(); // 2. 静态代码块(会被收集到<clinit>()中,按书写顺序执行) static { System.out.println("类初始化:执行静态代码块"); staticVar += 10; } // 静态变量初始化方法 private static int initStaticVar() { System.out.println("类初始化:执行静态变量赋值方法"); return 5; } // 3. 实例构造方法(<init>(),类初始化阶段不执行) public ClassInitDemo() { System.out.println("对象实例化:执行实例构造方法"); this.instanceVar = 20; } // 实例变量(与类初始化无关) private int instanceVar; public static void main(String[] args) { System.out.println("===== 开始执行main方法(触发类初始化) ====="); System.out.println("静态变量值:" + ClassInitDemo.staticVar); System.out.println("\n===== 创建对象(触发实例构造方法) ====="); new ClassInitDemo(); } }
执行结果

plaintext

类初始化:执行静态变量赋值方法 类初始化:执行静态代码块 ===== 开始执行main方法(触发类初始化) ===== 静态变量值:15 ===== 创建对象(触发实例构造方法) ===== 对象实例化:执行实例构造方法
结果分析
  1. 程序运行时,main方法调用ClassInitDemo.staticVar触发类的主动使用,进入类初始化阶段;
  2. 初始化阶段自动执行<clinit>()方法,按顺序执行「静态变量赋值」和「静态代码块」,输出前两行日志;
  3. 类初始化完成后,才执行main方法中的打印语句;
  4. 只有当执行new ClassInitDemo()时,才会调用实例构造方法<init>(),输出对象实例化日志,这一步与类加载的初始化阶段无关。
示例 2:验证<clinit>()仅执行一次

java

运行

public class ClinitOnceDemo { static { System.out.println("类初始化:<clinit>() 执行"); } public ClinitOnceDemo() { System.out.println("对象实例化:<init>() 执行"); } public static void main(String[] args) { // 第一次创建对象:先触发类初始化(<clinit>()执行),再执行<init>() new ClinitOnceDemo(); // 第二次创建对象:仅执行<init>(),<clinit>()不再执行 new ClinitOnceDemo(); // 第三次创建对象:仍仅执行<init>() new ClinitOnceDemo(); } }
执行结果

plaintext

类初始化:<clinit>() 执行 对象实例化:<init>() 执行 对象实例化:<init>() 执行 对象实例化:<init>() 执行
结果分析
  • 类的<clinit>()方法仅在第一次主动使用时执行一次,后续无论创建多少个对象,都不会再执行;
  • 实例构造方法<init>()每次创建对象时都会执行,与类初始化阶段无关。

四、补充:类初始化阶段的触发条件(主动使用)

只有满足「主动使用」条件时,JVM 才会触发类的初始化(执行<clinit>()),常见主动使用场景包括:

  1. 通过new关键字创建类的实例;
  2. 调用类的静态变量(非final修饰,final静态常量存入常量池,属于被动使用);
  3. 调用类的静态方法;
  4. 通过反射(Class.forName("类全限定名"))主动加载类;
  5. 初始化子类时,父类会被优先初始化(主动使用子类触发父类主动使用);
  6. 执行包含main()方法的主类(程序入口类,必然会被初始化)。

五、总结

  1. 类加载的初始化阶段不会执行实例构造方法(<init>(),即平时写的构造函数),仅执行类构造方法(<clinit>());
  2. 类构造方法<clinit>()是编译器自动生成的,用于初始化静态变量和静态代码块,仅执行一次;
  3. 实例构造方法<init>()用于初始化对象的实例资源,仅在创建对象(new/ 反射等)时执行,与类加载流程无关;
  4. 类初始化阶段由「主动使用」触发,核心任务是执行<clinit>(),保证类的静态资源完成初始化。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/9 23:48:45

小批量硅胶复模,医疗设备外壳3D打印定制,通过安全认证

小批量硅胶复模&#xff0c;医疗设备外壳3D打印定制&#xff0c;通过安全认证 做医疗设备研发的第3年&#xff0c;我终于在上海找到了靠谱的小批量硅胶复模和3D打印定制厂家——这是我踩了3次坑后&#xff0c;最真切的感受。 从事医疗设备研发行业的人都懂&#xff0c;新品研…

作者头像 李华
网站建设 2026/6/9 19:39:30

初次约会“社交算法”:高效对话框架让好感度指数级增长

一、需求分析与背景设定在情感社交场景中&#xff0c;初次约会可被视为一次关键的用户体验测试。许多技术从业者面临一个共同痛点&#xff1a;在代码世界游刃有余&#xff0c;但在面对面的情感交流中却常常陷入“系统异常”。本文将从结构化思维出发&#xff0c;为你构建一套高…

作者头像 李华
网站建设 2026/6/8 4:43:02

进度跟踪模块 Cordova 与 OpenHarmony 混合开发实战

&#x1f4cc; 概述 进度跟踪模块允许用户跟踪目标的完成进度。该模块集成了 Cordova 框架与 OpenHarmony 原生能力&#xff0c;提供了完整的进度更新和可视化展示。用户可以查看目标的当前进度、剩余时间和完成预测。模块支持进度的快速更新和历史记录查看。 &#x1f517; 完…

作者头像 李华
网站建设 2026/6/7 5:51:27

算法学习02|单调队列(上)学习总结

依旧是学习左神的课程&#xff1a;单调队列&#xff08; 上&#xff09; 单调队列的定义 单调队列&#xff0c;顾名思义&#xff0c;在实现一个双端队列&#xff08;队头队尾都可以插入、弹出元素&#xff09;的基础上&#xff0c;保持队列的数据从大到小&#xff08;从小到大…

作者头像 李华
网站建设 2026/6/9 9:47:10

Java并发编程基础:从线程管理到高并发应用实践

1. 理解线程&#xff1a;多任务执行的基石 1.1 什么是线程&#xff1f; 在现代操作系统中&#xff0c;进程是资源分配的基本单位&#xff0c;而线程是CPU调度的最小单位。可以把进程想象成一家公司&#xff0c;线程就是公司里的员工。 /** * 演示Java程序天生就是多线程程序 …

作者头像 李华