Java对象内存分配发生在堆中,引用存于栈内;new创建时JVM检查类加载、分配内存(指针碰撞或空闲列表)、初始化零值、设置对象头并执行构造函数;多线程下通过CAS同步或TLAB避免竞争;对象优先在新生代Eden区分配,经GC存活后移至Survivor区,达年龄阈值(默认15)则晋升老年代,大对象可直接进入老年代;通过-Xms、-Xmx、-XX:MaxTenuringThreshold等参数调优,提升性能。

Java中对象的内存分配主要发生在堆(Heap)上,由JVM自动管理。当使用new关键字创建对象时,JVM会在堆中为该对象分配内存空间,并返回指向该对象的引用。这个引用通常存储在栈(Stack)中,用于后续访问对象的数据和方法。
对象创建过程中的内存分配步骤
在执行new MyClass()时,JVM会经历以下几个关键阶段:
- 类加载检查:JVM先检查该类是否已被加载、解析和初始化,如果没有,则先执行类加载过程。
- 内存分配:在堆中划分一块确定大小的区域。分配方式主要有两种:指针碰撞(适用于规整内存,如Serial、ParNew收集器)和空闲列表(适用于内存碎片化情况,如CMS)。
- 初始化零值:分配完成后,系统将对象的实例变量初始化为默认值(如0、null、false等)。
- 设置对象头:包括哈希码、GC分代年龄、锁状态标志、类型指针等信息。
-
执行构造函数:调用
方法,完成程序员定义的初始化逻辑。
内存分配的线程安全性问题
多个线程同时创建对象时,可能会出现内存分配冲突。JVM提供两种解决方案:
- 同步处理:使用CAS(Compare-And-Swap)操作保证指针更新的原子性,配合失败重试机制确保分配成功。
- 本地线程分配缓冲(TLAB):每个线程在Java堆中预先分配一小块私有内存区域,称为TLAB(Thread Local Allocation Buffer)。大多数情况下,对象在本线程的TLAB中分配,避免了竞争,提高效率。只有当TLAB不足时才会进行同步分配。
堆内存结构与对象分布
现代JVM的堆通常分为新生代和老年代,对象根据生命周期被分配到不同区域:
系统特点: 商品多级分类检索、搜索,支持同一商品多重分类,自由设置显示式样 自由设置会员类型,自由设置权限项目,自由分配每种会员类型和每个会员的权限 灵活的商品定价,最多12级价格自由分配给各种会员类型或会员,也可针对单会员单商品特殊定价 强大的会员管理、帐户管理、订单管理功能和一系列帐务查询统计功能 灵活的会员积分系统,自由设置每个积分事件的积分计算方法 灵活的网站内容发布、管理系统,每个栏目可
立即学习“Java免费学习笔记(深入)”;
- 新生代(Young Generation):大多数对象在这里诞生和消亡。又细分为Eden区和两个Survivor区(S0、S1)。对象优先在Eden区分配,经历一次Minor GC后存活的对象会被移到Survivor区。
-
老年代(Old Generation):长期存活或大对象直接进入老年代。大对象(如长数组或字符串)可通过
-XX:PretenureSizeThreshold参数设置直接分配到老年代,避免频繁复制。 - 对象晋升机制:对象在Survivor区每经历一次GC,年龄加1,达到一定阈值(默认15)后进入老年代。
JVM优化与参数调优
合理配置JVM参数可以提升对象分配效率和系统性能:
-
-Xms和-Xmx设置堆的初始和最大大小,避免频繁扩容。 -
-XX:+UseTLAB启用线程本地分配缓冲,默认开启。 -
-XX:MaxTenuringThreshold控制对象晋升老年代的年龄阈值。 -
-XX:PretenureSizeThreshold指定多大的对象直接进入老年代。
基本上就这些。Java对象的内存分配是自动且高效的,理解其底层机制有助于编写高性能代码并进行有效调优。不复杂但容易忽略细节。









