Java虚拟机内存模型

Java运行时的数据区域分为线程共享区域和线程隔离的数据区域:

  • 线程共享:方法区、堆、执行引擎、本地库接口
  • 线程隔离:虚拟机栈、本地方法栈、程序计数器

方法区:

    存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码数据。运行时常量区也是方法区的一部分,运行时也能将新的常量放入池中,如String.intern()。

    因为HotSpot虚拟机把GC分代收集扩展至方法区,所以对于HotSpot虚拟机,习惯把方法区称为永久代(对于其他虚拟机不存在该概念)。因为永久代有MaxPermSize上线,用其来实现方法区容易遇到内存溢出。在JDK1.7后,HotSpot用Native Memory来实现方法区,开始放弃放久代,实例划的字符串不再分配到永久代,而是分配到堆的主要部分(年轻代和老年代)。在JDK1.8中,HotSopt移除永久代,用MetaSpace(元空间)取代,MetaSpace默认是没有上限的,只受本地可用的内存限制。

    方法区垃圾回收

  • 常量池回收
  1. 当没有地方引用常量则回收。
  • 类型回收(对类型的卸载)
  1. 类的所有实例已经回收
  2. 类的classloader已经回收
  3. 类的class对象没有被引用且无法通过反射访问该类

 

堆:

   堆区域的唯一目的是存放对象实例(Java虚拟机规范所有对象实例以及数据都要在堆上分配)。

   堆是垃圾收集器(GC)的主要管理区域,现在收集器基本都采用分代收集算法,导致堆细分为新生代和老年代,还可以细分为Eden空间、From Survivor空间、To Survivor空间。从内存分配角度,堆中可能划分出多个线程私有的分配缓冲区(TLAB)。无论怎么划分,存储的仍是对象实例,划分的目的是为了更好的分配或回收内存。

虚拟机栈:

  是Java方法执行的内存模型,每个方法执行的时候都会创建一个栈帧,用于存储

  • 局部变量
  • 操作数栈
  • 动态连接
  • 方法出口信息

本地方法栈:

  为虚拟机中用到的Native方法服务,由具体虚拟机自由实现。

程序计数器:

  是当前线程所执行字节码的行号指示器,记录的是正在执行的虚拟机字节码指令的地址,如果正在执行的Native方法,则值为空。这个内存区域是Java虚拟机中唯一不会内存溢出的区域。

直接内存:

  不是虚拟机运行时数据区的一部分,也不是Java虚拟机规范中定义的内存区域。如NIO可以直接使用Native函数库直接分配堆外内存,通过存储在Java堆中的DirectByteBuffer对象作为这个内存的引用进行操作。

 

 

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值