Java虚拟机(JVM)是一种在计算机上运行的虚拟机,它负责将Java字节码转换为可执行代码。在这个过程中,JVM需要管理内存空间,其中包括堆空间和栈空间。
堆空间是JVM中用于存储对象实例的内存区域。从JDK1.8开始,堆被划分为三个部分:新生代、老年代和永久代/元空间。当JVM启动时,它会自动为堆进行一次初始大小分配。如果堆的大小不够用,JVM将会自动扩容。堆空间的大小可以通过JVM启动参数-Xms和-Xmx来指定。
新生代是堆中的一个部分,用于存储新创建的对象实例。它又被分为Eden区和两个Survivor区(通常称为From区和To区)。当程序创建一个新的对象时,它被分配到Eden区。在Eden区满了之后,JVM会触发一次垃圾回收(GC)操作,清除无用的对象,并将还有用的对象移动到Survivor区中的一个区域中。在下一次GC操作中,位于From区的所有存活对象都会被移动到To区中,From区会被清空用来作为下一次GC的To区。在多次GC后,还存活的对象会被移动到老年代中,以避免频繁的GC操作。
老年代是堆中用于存储长期存活对象的内存区域。通常来说,老年代的容量要比新生代大。因为老年代的垃圾回收相对较少,所以当老年代满了之后,它将使用Full GC操作来释放无用的对象。
同时,从JDK1.8开始,JVM将永久代(PermGen)改为了元空间(Metaspace),这是一种基于本地内存的机制。元空间存储的是类定义和常量池等元数据信息。与永久代不同,元空间不再固定了大小,而是通过JVM选项-MetaspaceSize和-MaxMetaspaceSize指定上限。
除了堆外,JVM还有栈空间,栈空间用于存储程序执行时的方法调用和局部变量。每个线程都有自己的栈空间,它被分配在线程启动时。当一个方法被调用时,就会在栈空间中创建一个帧(Frame)用于存储方法的返回地址、参数、局部变量和操作数栈等信息。当方法执行完成时,它的帧就会被弹出栈空间。
总之,JVM的内存管理涉及到堆空间和栈空间。堆空间用于存储对象实例,而栈空间用于存储方法调用和局部变量。正确的内存管理对于Java程序的性能和稳定性具有非常重要的影响。开发人员可以通过监视JVM的内存使用情况来诊断和优化Java应用程序。