JVM内存模型

本文主要参考《Java虚拟机规范》并围绕比较主流的HotSpot虚拟机来研究

JVM的内存规范分为:堆,虚拟机栈,方法区,本地方法栈,程序计数器

一、堆

1.1 简介:

  • 线程共享
  • 分为新生代(Eden区和两个Survivor区From,To,默认比例为8:1:1)、老年代(Tenured)、永久代(Permanent)(注:Permanent在逻辑上也叫方法区,并也仅是在逻辑上属于堆的一部分)
  • 堆内存由GC管理,是垃圾回收的主要区域
  • 会出现内存溢出(OutOfMemoryError)

1.2 存储内容:

  • 对象
  • 数组

二、虚拟机栈

2.1 简介:

  • 线程私有
  • 每个方法就是一个栈帧
  • 栈帧结构为:局部变量表(基本数据类型、对象引用、returnAddress)、操作数栈(用于执行数据顺序)、动态链接(将符号引用转换为直接引用)、方法出口信息(用于退出方法)
  • 栈的内存大小可以固定,也可以扩展
  • 当栈的深度超过JVM规定的深度就会抛出StackOverFlowError
  • 会出现内存溢出(OutOfMemoryError)

2.2 存储内容:

  • 基本数据类型
  • 对象引用

三、方法区

3.1 简介:

  • 线程共享
  • 在逻辑上是堆中的一个逻辑部分,实际上不是必须的包含关系
  • 方法区只是逻辑上的称呼,并不是真实名称(1.8之前叫永久代、1.8叫元空间)

3.2 存储内容:

  • 类的信息(类型信息、域信息、方法信息)
  • 运行时常量池(-127~128缓存数值、符号引用、常量、(静态变量、静态常量、字符串值这三个1.7之前在运行时常量池,1.7在堆中))

四、本地方法栈

4.1 简介:

  • 线程私有
  • 为执行Native方法所服务的内存
  • 每个Native方法就是一个栈帧
  • 栈帧结构为:局部变量表(基本数据类型、对象引用、returnAddress)、操作数栈(用于执行数据顺序)、动态链接(将符号引用转换为直接引用)、方法出口信息(用于退出方法)
  • 栈的内存大小可以固定,也可以扩展
  • 当栈的深度超过JVM规定的深度就会抛出StackOverFlowError
  • 会出现内存溢出(OutOfMemoryError)

4.2 存储内容:

  • 基本数据类型
  • 对象引用

4.3 本地方法栈和虚拟机栈的区别:

主要的区别是:本地方法栈用于执行Native方法,虚拟机栈用于执行Java方法

五、程序计数器

5.1 简介:

  • 线程私有
  • 负责指向当前执行的字节码地址,保证线程切换回来能继续执行
  • 唯一不会出现内存溢出(OutOfMemoryError)的内存区域

5.2 存储内容:

  • 当前执行指令的地址

六、直接内存(堆外内存)

6.1 简介:

  • 非运行时数据区的一部分(本地内存的一部分)
  • 会出现内存溢出(OutOfMemoryError)

6.2 存储内容:

  • 堆内内存不足或数据过大时,帮助堆存储数据

七、JDK1.7

八、JDK1.8

九、1.7和1.8区别

主要的区别就是:1.8将方法区由元空间来实现,元空间是在本地内存

十、为什么要将永久代分出去?

  • 永久代的空间大小无法确定
  • 为了减少FullGC

十一、为什么要将StringTable放在Heap中

因为方法区只有FullGC的时候才会回收

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值