参考资料:深入理解Java虚拟机:JVM高级特性与最佳实践(第3版)周志明
Java 虚拟机在执行 Java 程序的过程中会把它所管理的内存划分为若干个不同的数据区域。

字节码解释器工作时通过改变这个计数器的值,选取下一条执行的字节码指令。此内存区域是唯一一个在《Java 虚拟机规范》中没有规定任何OutOfMemoryError 情况的区域。
它的生命周期和线程相同,虚拟机栈描述的是Java方法执行的线程内存模型。
每个方法执行时,虚拟机都会创建一个栈帧(Stack Frame)用于存储:
每个方法从调用到结束,就对应一个栈帧在虚拟机栈从出栈到入栈的过程。
关于局部变量表:
这个内存区域规定了两类异常情况:
1、如果线程请求的栈深度大于虚拟机所允许的深度,将抛出
StackOverflowError异常2、如果Java虚拟机栈的容量可以动态拓展(HotSpot不可以,以前的Classic可以),当栈拓展到无法申请到足够的内存,会抛出
OutOfMemoryError异常
本地方法栈与虚拟机栈发挥作用是相似的,区别在于本地方法栈是为虚拟机使用到的本地(Native)方法服务。
与虚拟机栈一样,本地方法栈也会在栈深度溢出或者栈扩展失败时分别抛出 StackOverflowError 和OutOfMemoryError 异常。
Java堆(Java Heap)是虚拟机管理的内存中最大的一块,是被所有线程共享的一块内存区域。
JavaHeap的唯一目的就是存放对象实例,几乎所有的对象实例都在这分配内存
在Java 7之前,字符串常量池部分地位于方法区。但从Java 7开始,字符串常量池被转移到了Java堆内存中,这是为了便于垃圾回收和内存分配。
字符串常量池中存放的什么?
String str1 = "hello" String str2 = new String("world"); String str3 = "hello"; new String("world") 会在Java堆中创建一个新的字符串对象。方法区(Method Area)用于存储已被虚拟机加载的类型信息、常量、静态变量、即时编译器编译后代码缓存的数据。
在JDK8之前,HotSpot JVM将方法区实现为永久代,但这部分内存是有固定大小的,并非真正的"永久",到了JDK8废弃了永久代的概念,在本地内存中实现元空间来代替。
这种变化解决了永久代空间固定可能导致的内存溢出问题
《Java 虚拟机规范》对方法区的约束是非常宽松的,除了和 Java 堆一样不需要连续的内存和可以选择固定大小或者可扩展外,甚至还可以选择不实现垃圾收集。
暂时无法在飞书文档外展示此内容
java语言运行时也可以将新的常量放入常量池中,这种特性用的比较多的是String的intern()方法,将常量放入了字符串常量池。
String str1 = new String("Hello"); String str2 = str1.intern(); String str3 = "Hello"; System.out.println(str1 == str2); // 输出 false System.out.println(str2 == str3); // 输出 true DirectByteBuffer对象作为这块内存的引用进行操作。 这样能显著提高性能。java.lang.OutOfMemoryError: Java heap space ------>java堆内存溢出,此种情况最常见见,一般由于内存泄露或者堆的大小设置不当引起。对于内存泄露,需要通过内存监控软件查找程序中的泄露代码,而堆大小可以通过虚拟机参数-Xms,-Xmx等修改。
java.lang.OutOfMemoryError: PermGen space ------>java永久代溢出,即方法区溢出了,一般出现于大量Class或者jsp页面,或者采用cglib等反射机制的情况,因为上述情况会产生大量的Class信息存储于方法区。此种情况可以通过更改方法区的大小来解决,使用类似-XX:PermSize=64m-XX:MaxPermSize=256m的形式修改。
java.lang.StackOverflowError ------>不会抛OOM Error,但也是比较常见的Java内存溢出。JAVA虚拟机栈溢出,一般是由于程序中存在死循环或者深度递归调用造成的,栈大小设置太小也会出现此种溢出。可以通过虚拟机参数-Xss来设置栈的大小。