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

Java里的堆和栈,本质是JVM内存中两个职责明确、行为迥异的区域:堆专管对象生命周期,栈专管方法执行流程。
堆是JVM启动时创建的唯一一块共享内存区域,被所有线程共用。用new关键字创建的每一个对象(包括数组),都必须落在堆中。比如:
String s = new String("hello"); → 字符串对象"hello"存于堆,s只是栈里的一个引用List<integer> list = new ArrayList();</integer> → ArrayList实例本身在堆,list变量在栈堆不关心谁在用它,只负责分配空间和配合GC回收。对象是否存活,取决于有没有栈、静态区或其它堆对象还在引用它——这就是GC Roots追踪的起点。堆大小可通过-Xms和-Xmx参数调整,空间不足会直接抛出OutOfMemoryError。
每个线程启动时,JVM都会为其分配独立的虚拟机栈,里面按调用顺序压入一个个栈帧。每个栈帧对应一次方法调用,保存着:
立即学习“Java免费学习笔记(深入)”;
方法开始执行即入栈,执行结束立即出栈,变量随栈帧销毁而自动释放。例如int a = 5;、Object obj;这类定义都在栈里,但obj本身不存对象,只存指向堆中对象的地址。栈空间小、速度快,但大小固定;深度超限会触发StackOverflowError(如无限递归)。
从四个维度看差异:
OutOfMemoryError: Java heap space;栈溢出报StackOverflowError
JVM里的“堆”和“栈”是内存管理概念,和算法课讲的二叉堆(用于优先队列)、后进先出栈(LIFO结构)没有实现关系。JVM栈的底层实现确实类似数据结构栈(压栈/出栈),但JVM堆的组织方式更接近空闲链表+分代管理,跟完全二叉树毫无关联。看到“堆内存”,就只想到new出来的对象;看到“栈内存”,就只想到方法调用链和局部变量。
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号