登录
首页 >  文章 >  java教程

Java new 创建对象过程与内存分配解析

时间:2026-04-01 13:36:24 219浏览 收藏

Java中new创建对象远非简单的内存分配,而是一个由JVM深度协同完成的精密流程:从类加载检查(确保类已加载、验证、准备、解析与初始化)、到堆内存的高效分配(依托TLAB优化多线程并发,辅以指针碰撞或空闲列表策略),再到零值初始化、对象头设置及构造方法执行,最终完成引用赋值——每一步都关乎性能、安全与语义正确性。理解这一全过程,是掌握JVM底层机制、优化对象创建开销、排查内存问题的关键入口。

Java new 运算符创建对象的过程与内存分配

Java 中 new 运算符创建对象的过程,本质上是 JVM 协调类加载、内存分配、初始化等多个步骤的完整流程,不是简单“分配一块内存”这么轻量。

类加载检查(Class Loading Check)

执行 new 指令时,JVM 首先检查该指令的参数是否能在常量池中定位到一个类的符号引用;然后确认该类是否已被加载、解析和初始化。若未完成,则触发类加载过程(加载 → 验证 → 准备 → 解析 → 初始化)。

例如:new ArrayList() 会确保 java.util.ArrayList 类已加载并完成初始化(包括静态代码块和静态变量赋值)。

内存分配(Memory Allocation)

类加载检查通过后,JVM 在堆中为新对象分配内存。分配方式取决于垃圾收集器和堆内存布局:

  • 指针碰撞(Bump the Pointer):适用于内存规整的场景(如 Serial、ParNew 收集器配合 Serial Old)。JVM 维护一个“空闲指针”,分配即移动指针位置,高效且原子。
  • 空闲列表(Free List):适用于内存不连续的场景(如 CMS、G1 的部分区域)。JVM 维护一个记录可用内存块的列表,分配时从中查找合适大小的空间。
  • TLAB(Thread Local Allocation Buffer):为避免多线程并发分配时加锁,JVM 默认为每个线程在 Eden 区预分配一小块私有缓冲区。大多数对象优先在 TLAB 中分配,失败再尝试 Eden 共享区或触发 GC。

对象初始化(Zeroing & Initialization)

内存分配完成后,JVM 立即将该内存空间初始化为零值(不包括对象头),保证实例字段即使未显式赋值也具备默认值(如 int=0、reference=null、boolean=false)。

紧接着设置对象头信息,包括:

  • 哈希码(延迟计算,首次调用 hashCode() 时生成)
  • GC 分代年龄
  • 锁状态标志、线程持有的锁等
  • 类型指针(指向方法区中的类元数据,用于确定对象所属类)

最后,JVM 调用类的构造方法( 方法),按 Java 代码逻辑对字段进行显式赋值、执行实例初始化块和构造器语句,使对象进入可使用状态。

对象引用赋值(Reference Assignment)

构造方法执行完毕,new 表达式返回对象的引用(即地址),赋值给栈帧中的局部变量或字段。注意:此时对象才真正“诞生”,对外可见。

例如:String s = new String("hello"); 中,s 存储的是堆中该 String 实例的引用,而非对象本身。

今天关于《Java new 创建对象过程与内存分配解析》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>