JVM(Java虚拟机)的内存模型是理解Java程序运行时内存分配和管理的关键。JVM的内存模型主要分为以下几个部分:
1. 程序计数器(Program Counter Register)
- 作用:记录当前线程所执行的字节码指令的地址。
- 特点:每个线程都有一个独立的程序计数器,是线程私有的。
2. Java虚拟机栈(Java Virtual Machine Stack)
- 作用:存储方法的局部变量、操作数栈、动态链接、方法出口等信息。
- 特点:每个线程都有一个私有的虚拟机栈,每个方法被调用时都会创建一个栈帧(Frame),方法执行完毕后栈帧被销毁。
3. 本地方法栈(Native Method Stack)
- 作用:与虚拟机栈类似,但用于支持本地方法(如JNI调用)。
- 特点:每个线程都有一个私有的本地方法栈。
4. 堆(Heap)
- 作用:存储对象实例和数组。
- 特点:是所有线程共享的内存区域,是垃圾回收的主要区域。堆可以细分为新生代(Young Generation)和老年代(Old Generation)。
5. 方法区(Method Area)
- 作用:存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
- 特点:是所有线程共享的内存区域。在HotSpot虚拟机中,方法区也被称为“永久代”(Permanent Generation),但在Java 8及以后版本中,永久代被元空间(Metaspace)取代。
6. 运行时常量池(Runtime Constant Pool)
- 作用:是方法区的一部分,存储编译期生成的各种字面量和符号引用。
- 特点:每个类或接口的运行时常量池都是唯一的。
7. 直接内存(Direct Memory)
- 作用:不是JVM运行时数据区的一部分,但也会被频繁使用,通过
java.nio包下的类进行分配。 - 特点:直接内存的分配不受JVM堆大小限制,但受本机内存限制。
总结
- 程序计数器:线程私有,记录字节码指令地址。
- 虚拟机栈:线程私有,存储方法的局部变量等信息。
- 本地方法栈:线程私有,支持本地方法。
- 堆:线程共享,存储对象实例和数组。
- 方法区:线程共享,存储类信息、常量、静态变量等。
- 运行时常量池:方法区的一部分,存储字面量和符号引用。
- 直接内存:非JVM运行时数据区,通过
java.nio包进行分配。