当前位置: 首页 > news >正文

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>(),保证类的静态资源完成初始化。
http://www.cnnetsun.cn/news/175383.html

相关文章:

  • STM32学习——AD单通道AD多通道
  • 基于Spring Boot的农产品销售系统的设计与实现毕设源码
  • 基于Spring Boot的流浪动物救助平台的设计与实现毕业设计
  • 备份恢复-Cordovaopenharmony本地安全方案
  • 创建目标模块 Cordova 与 OpenHarmony 混合开发实战
  • 解决MQ消息丢失问题的5种方案
  • 芜湖,千兆网络下载速率只有10MB秒,过的什么苦日子
  • AI一周大事盘点(2025年12月14日~2025年12月20日)
  • K3s + Sysbox:让容器拥有“虚拟机的灵魂”
  • 8 个降AI率工具推荐,继续教育学生必备
  • 从开发一个AI美女聊天群组开始
  • 12.2K Star 爆火!开源免费的 FileConverter:右键一键搞定音视频 / 图片 / 文档转换,告别多工具切换
  • Java毕设项目:基于springboot的养宠物指南服务平台系统的设计与实现(源码+文档,讲解、调试运行,定制等)
  • 10 个降AI率工具,继续教育学生高效避坑指南
  • Java毕设项目推荐-基于SpringBoot的演唱会门票在线预定系统的设计与实现基于springboot的演唱会购票系统的设计与实现【附源码+文档,调试定制服务】
  • 升压芯片很简单(一),快速选择升压芯片+利用升压芯片设计LED电源
  • 基于web的人才招聘网站设计 nodejs vue
  • 测试20个降AI率工具后,我找到了2个去ai痕迹效果好的网站,还有免费降AI额度。
  • Thinkphp和Laravel在线点餐系统的设计与实现vue
  • 现代cpp在传统内存分配上的改进
  • Java毕设项目:基于springboot的物业报修系统的设计与实现(源码+文档,讲解、调试运行,定制等)
  • 【计算机毕业设计案例】基于springboot的物业报修系统的设计与实现线上化的报修管理平台(程序+文档+讲解+定制)
  • Java毕设选题推荐:基于springboot的社区团购系统的设计与实现、拼团下单、配送调度、资金结算【附源码、mysql、文档、调试+代码讲解+全bao等】
  • Java计算机毕设之基于springboot的幼儿园管理系统的设计与实现为幼儿园(含普惠园、民办园、连锁园)设计的 “家园共育 + 日常运营 + 安全监管(完整前后端代码+说明文档+LW,调试定制等)
  • I/O多路复用
  • 视频播放器PotPlayer下载安装教程:超详细图文步骤(PC+安卓)
  • Semantic Kernel 实战系列(六) - Memory与向量存储
  • 一个基于 .NET MAUI 的开箱即用的 UI 组件库,可快速搭建面向业务的应用程序界面!
  • Semantic Kernel 实战系列(七) - 高级主题 - Agents 与多代理系统
  • LeetCode每日一题——K个一组翻转链表