Java中堆是线程共享、存放对象实例的内存区,由GC管理;栈是线程私有、存放局部变量和方法调用信息的内存区,随方法进出自动释放。

Java里的堆和栈,本质是JVM内存中两个职责明确、行为迥异的区域:堆专管对象生命周期,栈专管方法执行流程。
堆:所有对象的统一存放地
堆是JVM启动时创建的唯一一块共享内存区域,被所有线程共用。用new关键字创建的每一个对象(包括数组),都必须落在堆中。比如:
-
String s = new String("hello");→ 字符串对象"hello"存于堆,s只是栈里的一个引用 -
List→ ArrayList实例本身在堆,list变量在栈list = new ArrayList();
堆不关心谁在用它,只负责分配空间和配合GC回收。对象是否存活,取决于有没有栈、静态区或其它堆对象还在引用它——这就是GC Roots追踪的起点。堆大小可通过-Xms和-Xmx参数调整,空间不足会直接抛出OutOfMemoryError。
栈:每个线程独享的执行快照区
每个线程启动时,JVM都会为其分配独立的虚拟机栈,里面按调用顺序压入一个个栈帧。每个栈帧对应一次方法调用,保存着:
立即学习“Java免费学习笔记(深入)”;
- 局部变量表(基本类型值、对象引用)
- 操作数栈(字节码执行时的临时运算空间)
- 动态连接、方法返回地址等控制信息
方法开始执行即入栈,执行结束立即出栈,变量随栈帧销毁而自动释放。例如int a = 5;、Object obj;这类定义都在栈里,但obj本身不存对象,只存指向堆中对象的地址。栈空间小、速度快,但大小固定;深度超限会触发StackOverflowError(如无限递归)。
关键区别一目了然
从四个维度看差异:
- 归属:堆是线程共享的;栈是线程私有的
- 存什么:堆存对象实体和数组;栈存局部变量、引用、方法现场
- 生命周期:堆对象由GC决定生死;栈变量随方法进出自动生灭
-
异常表现:堆满报
OutOfMemoryError: Java heap space;栈溢出报StackOverflowError
别混淆:堆/栈 ≠ 数据结构中的堆/栈
JVM里的“堆”和“栈”是内存管理概念,和算法课讲的二叉堆(用于优先队列)、后进先出栈(LIFO结构)没有实现关系。JVM栈的底层实现确实类似数据结构栈(压栈/出栈),但JVM堆的组织方式更接近空闲链表+分代管理,跟完全二叉树毫无关联。看到“堆内存”,就只想到new出来的对象;看到“栈内存”,就只想到方法调用链和局部变量。










