当前位置: 首页 > 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.jsqmd.com/news/120595/

相关文章:

  • 3.1
  • 施工项目管理系统有哪些:10款支持进度、成本与质量全流程管控的工具
  • 多项目管理系统怎么选:8款支持跨项目资源与进度统筹的解决方案
  • 优化程序性能:JVM 会对final变量进行优化(如编译期常量折叠,直接将常量值嵌入字节码中,避免运行时获取);具体含义
  • 142_尚硅谷_数组的使用价值
  • 数字化项目管理系统分享:7款助力企业实现项目智能化协同的工具精选
  • 3.3
  • 财务目标页面 UI 与进度展示 - Cordova与OpenHarmony混合开发实战
  • 基于Spring Boot的美食信息分享平台设计与实现毕设源码
  • 2025年主流项目集管理系统工具推荐:6款助力企业实现战略级项目群管控的系统盘点
  • 基于Spring Boot的稀有植物保护平台开发毕设源码
  • 3.4
  • CordovaOpenHarmony使用指南文档
  • 配置Wireshark抓取https数据包
  • C#应用程序取得当前目录和退出
  • CF1088E Ehab and a component choosing problem
  • 【文章记录-001】
  • 9.37万亿元与9.58亿用户:拆解2025上半年中国数字消费的增长引擎
  • 如何在 Power BI 中基于多个列应用 RLS
  • 报表生成功能Cordova与OpenHarmony混合开发实战
  • 3.7
  • 3.8
  • ARGB和对其原理是什么
  • 初次约会“社交算法”:高效对话框架让好感度指数级增长
  • Cordova与OpenHarmony全文搜索功能
  • EE308FZ_Fifth Assignment_Alpha Sprint_Sprint Essay2
  • FFT:嵌入式开发的“算力引擎”,支持Q15定点和F32浮点两种算法
  • DeepSeek vs 通义千问:代码生成场景准确率与响应速度实测分析报告
  • 第七届传智杯 初赛 小红的四子棋 题解 简单bfs遍历
  • 碎片化学习:DeepSeek 定制个人技术成长计划与知识点拆解