Java内存分为栈和堆:栈存局部变量与方法调用信息,线程私有、自动管理、LIFO回收;堆存new对象,线程共享、GC管理;栈中存引用,堆中存对象实体。

Java程序运行时的内存主要分为栈(Stack)和堆(Heap)两大部分,它们分工明确、生命周期不同,理解二者区别是排查内存问题和写出高效代码的基础。
栈内存:存放局部变量和方法调用信息
栈内存由JVM自动管理,每个线程拥有独立的栈空间。每当一个方法被调用,JVM就在当前线程栈中压入一个“栈帧”(Stack Frame),里面包含该方法的参数、局部变量、操作数栈和返回地址等。
- 基本数据类型(如int、boolean、char)和对象引用(reference)都直接存放在栈中
- 栈内存分配和释放极快,遵循后进先出(LIFO)原则,方法执行结束,对应栈帧自动弹出,内存立即回收
- 栈空间有限,深度过大的递归或过多的局部变量可能导致StackOverflowError
堆内存:存放所有new出来的对象实例
堆是JVM中最大的一块共享内存区域,所有线程共用。使用new关键字创建的对象、数组,以及其内部字段(包括引用类型的字段指向的对象)都分配在堆上。
- 对象本身永远在堆中,哪怕它是一个局部变量所引用的对象
- 堆内存由垃圾收集器(GC)管理,对象是否存活由可达性分析决定;不再被引用的对象会在合适时机被回收
- 堆空间不足且无法扩展时,会抛出OutOfMemoryError: Java heap space
栈与堆的协作关系
栈和堆不是割裂的,而是紧密配合:栈中保存的是“谁在用”,堆中保存的是“用的是什么”。例如:
立即学习“Java免费学习笔记(深入)”;
String s = new String("hello");
- 变量s(引用)存在栈中
- 字符串对象"hello"实际存储在堆中(字符串常量池也属于堆的一部分,JDK 7+起)
- 如果另一个方法里也有String t = s;,只是把栈中的引用值复制过去,两个变量指向堆中同一个对象
容易混淆的几个点
- static变量和类信息存放在方法区(Metaspace,JDK 8+),不属于堆也不属于栈,但常被误认为在堆中
- 局部变量如果是基本类型,值就在栈;如果是引用类型,引用在栈,对象在堆
- 逃逸分析可能让JVM将某些本该在堆上的小对象优化到栈上分配(标量替换),但这对开发者透明,无需手动干预










