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

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

类加载阶段:从字节码到内存中的Class对象

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

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

  • 显式触发加载:用 Class.forName("com.example.MyClass")(默认初始化),或 ClassLoader.loadClass("com.example.MyClass")(不初始化)
  • 加载失败典型错误:ClassNotFoundException(路径/拼写错)、NoClassDefFoundError(已加载过但链接时找不到依赖类)
  • 自定义类加载器需重写 findClass(String name),而非 loadClass(String name),否则破坏委派机制

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

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

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

立即学习“Java免费学习笔记(深入)”;

  • final static 基本类型字段在准备阶段就赋值(编译期常量),不会等到初始化
  • 接口也有初始化阶段,但只有当首次调用其 default 方法或静态方法时才触发(JDK 8+)
  • 子类初始化前,父类一定已完成初始化;但通过子类引用父类静态字段(非 final)只会触发父类初始化,不触发子类

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

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

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

  • 排查类泄漏:用 jcmd VM.native_memory summaryjstat -gc 观察 metaspace 使用趋势
  • JDK 9+ 中,模块化系统允许更细粒度的类加载器隔离,但卸载逻辑未变
  • Instrumentation.redefineClasses() 可热替换类定义,但仅限方法体变更,不能增删字段或改变继承关系

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

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

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

  • 动态代理生成的类(如 $Proxy0)由 sun.misc.Launcher$AppClassLoader 的子类加载,其卸载依赖于代理创建者的生命周期
  • Spring 的 ContextClassLoader 切换可能导致意外的类加载器层级,影响资源查找和类可见性
  • 模块路径(--module-path)下类的加载由 ModuleLayer 管理,其生命周期与模块图绑定,比传统 classpath 更严格


# java  # js  # bootstrap  # app  # 字节  # ssl  #   # proxy  # 动态代理  # java类  # red 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Java Adapter 适配器模式(类适配器,对象适配器)优缺点对比  Gemini怎么用新功能实时问答_Gemini实时问答使用【步骤】  深圳网站制作设计招聘,关于服装设计的流行趋势,哪里的资料比较全面?  如何利用DOS批处理实现定时关机操作详解  laravel怎么在请求结束后执行任务(Terminable Middleware)_laravel Terminable Middleware请求结束任务执行方法  如何在局域网内绑定自建网站域名?  Laravel如何使用Scope本地作用域_Laravel模型常用查询逻辑封装技巧【手册】  如何用wdcp快速搭建高效网站?  如何在建站之星网店版论坛获取技术支持?  Python3.6正式版新特性预览  百度浏览器如何管理插件 百度浏览器插件管理方法  ChatGPT常用指令模板大全 新手快速上手的万能Prompt合集  如何基于云服务器快速搭建网站及云盘系统?  Laravel如何使用缓存系统提升性能_Laravel缓存驱动和应用优化方案  如何用西部建站助手快速创建专业网站?  海南网站制作公司有哪些,海口网是哪家的?  php打包exe后无法访问网络共享_共享权限设置方法【教程】  敲碗10年!Mac系列传将迎来「触控与联网」双革新  Android滚轮选择时间控件使用详解  香港网站服务器数量如何影响SEO优化效果?  php增删改查怎么学_零基础入门php数据库操作必知基础【教程】  微信小程序 五星评分(包括半颗星评分)实例代码  千问怎样用提示词获取健康建议_千问健康类提示词注意事项【指南】  如何注册花生壳免费域名并搭建个人网站?  Laravel如何创建和注册中间件_Laravel中间件编写与应用流程  Laravel如何实现模型的全局作用域?(Global Scope示例)  Laravel如何实现密码重置功能_Laravel密码找回与重置流程  Win11搜索不到蓝牙耳机怎么办 Win11蓝牙驱动更新修复【详解】  如何快速搭建安全的FTP站点?  ChatGPT回答中断怎么办 引导AI继续输出完整内容的方法  javascript日期怎么处理_如何格式化输出  Swift中循环语句中的转移语句 break 和 continue  Laravel软删除怎么实现_Laravel Eloquent SoftDeletes功能使用教程  中山网站推广排名,中山信息港登录入口?  jQuery validate插件功能与用法详解  高性价比服务器租赁——企业级配置与24小时运维服务  安克发布新款氮化镓充电宝:体积缩小 30%,支持 200W 输出  长沙企业网站制作哪家好,长沙水业集团官方网站?  Java类加载基本过程详细介绍  Laravel如何处理JSON字段_Eloquent原生JSON字段类型操作教程  如何快速搭建自助建站会员专属系统?  家族网站制作贴纸教程视频,用豆子做粘帖画怎么制作?  Gemini手机端怎么发图片_Gemini手机端发图方法【步骤】  如何在IIS中配置站点IP、端口及主机头?  JavaScript如何实现继承_有哪些常用方法  node.js报错:Cannot find module 'ejs'的解决办法  Android中Textview和图片同行显示(文字超出用省略号,图片自动靠右边)  javascript中数组(Array)对象和字符串(String)对象的常用方法总结  Laravel怎么实现API接口鉴权_Laravel Sanctum令牌生成与请求验证【教程】  黑客如何利用漏洞与弱口令入侵网站服务器?