理解永久代、元空间、JVM内存结构!

Java虚拟机(JVM)的内存结构主要包括以下几个部分:

1. 堆(Heap):堆是JVM中最大的一块内存区域,主要用于存储对象实例和数组。堆被所有线程共享,是垃圾收集器主要管理的区域。

2. 方法区(Method Area):方法区也是被所有线程共享的内存区域,用于存储已被虚拟机加载的类信息、常量、静态变量等数据。(JDK1.7 之前,字符串常量池存放在永久代。JDK1.7 字符串常量池和静态变量从永久代移动了 Java 堆中。)

3. 虚拟机栈(JVM Stack):每个线程有一个私有的JVM栈,它的生命周期与线程相同。JVM栈描述的是Java方法执行的内存模型:每个方法在执行的同时都会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息。

4. 本地方法栈(Native Method Stack):本地方法栈与虚拟机栈所发挥的作用是非常相似的,其区别不过是虚拟机栈为虚拟机执行Java方法(也就是字节码)服务,而本地方法栈则为虚拟机使用到的Native方法服务。

5. 程序计数器(Program Counter Register):程序计数器是一块较小的内存空间,它的作用可以看做是当前线程所执行的字节码的行号指示器。

6. 直接内存(Direct Memory):直接内存并不是虚拟机运行时数据区的一部分,也不是Java虚拟机规范中定义的内存区域。它是在JDK 1.4中新加入的NIO类,引入了一种基于通道(Channel)与缓冲区(Buffer)的I/O方式,它可以使用native函数库直接分配堆外内存,然后通过一个存储在Java堆中的DirectByteBuffer对象作为这块内存的引用进行操作。这样能在一些场景中显著提高性能,因为避免了在Java堆和Native堆中来回复制数据。 

JDK1.7为什么将字符串常量池移动到堆中?

主要是因为永久代(方法区实现)的GC回收效率太低,只有在整堆(Full GC)的时候才会被执行GC。Java程序中通常会有大量的被创建的字符串等待回收,将字符串常量池放到堆中,能够更高效地回收字符串内存。

方法区和永久代以及元空间关系? 

 方法区和永久代以及元空间的关系很像 Java 中接口和类的关系,类实现了接口,这里的类就可以看作是永久代和元空间,接口可以看作是方法区,也就是说永久代以及元空间是 HotSpot 虚拟机对虚拟机规范中方法区的两种实现方式。并且,永久代是 JDK 1.8 之前的方法区实现,JDK 1.8 及以后方法区的实现变成了元空间。

为什么要将永久代(PermGen)替换为元空间(MetaSpace)?

1. PermGen有固定的大小,并且从Java堆中分配,如果PermGen空间大小不正确,可能会导致OutOfMemoryError。
2. PermGen用于存储类元数据、interned字符串和其他JVM内部数据。随着应用程序变得越来越复杂,PermGen空间很容易耗尽。
3. 相反,元空间是在Java堆之外分配的,这使得它可以根据应用程序的需求动态调整大小。
4. 元空间还受益于使用本机内存,与固定大小的PermGen空间相比,本机内存可以更高效和可扩展。
5. 此外,当PermGen空间满时,元空间通过消除需要进行完整垃圾回收的需求,提供更好的垃圾回收性能,无需进行完整垃圾回收。 

总之,JVM 1.8将PermGen替换为元空间是为了解决PermGen的限制,并为管理类元数据和其他JVM内部数据提供更灵活和高效的解决方案。 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值