JVM(Java 虚拟机)内存模型主要包括以下几个区域:
1. 程序计数器(Program Counter Register)
- 作用:是一块较小的内存空间,可以看作是当前线程所执行的字节码的行号指示器。
- 特点:线程私有,每个线程都有独立的程序计数器。此区域是 JVM 内存区域中唯一一个没有规定任何 OutOfMemoryError
情况的区域。
2. Java 虚拟机栈(Java Virtual Machine Stacks)
- 作用:用于存储方法调用的栈帧信息,包括局部变量表、操作数栈、动态链接、方法出口等。
- 特点:线程私有。如果线程请求的栈深度大于虚拟机所允许的深度,将抛出 StackOverflowError
异常;如果虚拟机栈可以动态扩展,当扩展时无法申请到足够的内存,会抛出 OutOfMemoryError
异常。
例如,一个方法不停地递归调用自身,可能会导致栈深度超出限制,抛出 `StackOverflowError` 。
3. 本地方法栈(Native Method Stacks)
- 作用:与 Java 虚拟机栈类似,但是它服务于 Native 方法。
- 特点:线程私有,可能抛出的异常与 Java 虚拟机栈相同。
4. Java 堆(Java Heap)
- 作用:是 JVM 管理的最大一块内存区域,用于存放对象实例和数组。
- 特点:所有线程共享。如果在堆中没有内存完成实例分配,并且堆也无法再扩展时,将会抛出 OutOfMemoryError
异常。
比如,大量创建对象而没有及时释放,可能导致堆内存不足。
5. 方法区(Method Area)
- 作用:存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
- 特点:线程共享。当方法区无法满足内存分配需求时,会抛出 OutOfMemoryError
异常。
在 JDK 8 之后,使用元空间替代了永久代,元空间使用的是本地内存。
6. 运行时常量池(Runtime Constant Pool)
- 作用:是方法区的一部分,用于存放编译期生成的各种字面量和符号引用。
- 特点:具备动态性,运行期间也可能将新的常量放入池中。
例如,String 类的 `intern` 方法就可能会将字符串常量放入运行时常量池中。