贝利信息

在Java中什么是类的生命周期_Java类加载到卸载解析

日期:2026-01-05 00:00 / 作者:P粉602998670
Java类生命周期始于加载,即ClassLoader将字节码解析为Class对象,遵循双亲委派模型;随后经历验证、准备、解析、初始化四步连接与初始化阶段;使用后仅在类、其ClassLoader及Class对象均被GC回收时才可能卸载。

类加载阶段:从字节码到内存

中的Class对象

Java类的生命周期始于加载,但“加载”不等于读取文件——它特指 ClassLoader.class 字节码解析为 java.lang.Class 实例的过程。这个过程由双亲委派模型控制,BootstrapClassLoaderExtensionClassLoaderAppClassLoader 逐级委托,避免重复加载和核心类被篡改。

常见误区是认为 new MyClass() 才触发加载;实际上,首次主动使用该类(如调用静态方法、访问静态字段、反射 Class.forName("MyClass"))才会触发,而 MyClass.class 或子类引用父类静态字段(非 final)也可能触发。

连接与初始化:验证、准备、解析、初始化四步不可跳过

加载之后不是直接执行代码,而是进入连接(Linking)阶段,分为验证、准备、解析三步,再执行初始化(Initialization)。这四步顺序固定,且只发生一次。

验证 检查字节码是否符合 JVM 规范(如栈溢出、非法跳转);准备 为类变量(static 字段)分配内存并设默认值(如 int 设为 0,String 设为 null);解析 将常量池中符号引用(如 "java/lang/Object")转为直接引用(内存地址);初始化 才真正执行 static 块和静态字段赋值语句。

使用与卸载:GC 回收 Class 对象的严格条件

类在初始化后进入“使用”阶段,即正常运行实例方法、静态方法等。但“卸载”极少发生——JVM 只有在满足三个条件时才可能卸载一个类:该类所有实例已被 GC;该类的 ClassLoader 实例已被 GC;该类的 Class 对象没有被任何地方引用(包括 JNI、线程栈、静态变量)。

这意味着:系统类(由 Bootstrap 类加载器加载)永不卸载;Web 应用热部署失败常因旧 ClassLoader 被持有(如线程局部变量、静态缓存、未关闭的 JDBC 驱动),导致其加载的所有类无法卸载,最终 OutOfMemoryError: Metaspace

类生命周期中容易被忽略的关键点

很多人以为类加载完就“稳定”了,其实类的生命周期高度依赖类加载器的生命周期,而不是类本身。同一个类名,被不同 ClassLoader 加载,就是两个完全无关的类——它们的实例不能互相转型,static 变量不共享,甚至 instanceof 都会返回 false

另一个盲区是“被动引用”不触发初始化:比如数组声明 MyClass[] arr = new MyClass[10] 不触发 MyClass 初始化;常量接口字段访问也不触发接口初始化(因为编译期已内联)。