Java内存模型

由于Java程序是交由JVM执行的,所以我们在谈Java内存区域划分的时候事实上是指JVM内存区域划分。

 

1.Java程序执行流程

如图所示

首先Java源代码文件(.java后缀)会被Java编译器编译为字节码文件(.class后缀),

然后由JVM中的类加载器加载各个类的字节码文件,

加载完毕之后,交由JVM执行引擎执行。

Java内存模型指的就是Runtime Data Area(运行时数据区),即程序执行期间用到的数据和相关信息保存区。

 

2.Java内存模型

根据 JVM 规范,JVM 内存共分为虚拟机栈、堆、方法区、程序计数器、本地方法栈五个部分。结构如下图:

  2.1.PC程序计数器:

  • 每个线程对应有一个程序计数器。
  • 各线程的程序计数器是线程私有的,互不影响,是线程安全的。
  • 程序计数器记录线程正在执行的内存地址,以便被中断线程恢复执行时再次按照中断时的指令地址继续执行

2.2.Java栈JavaStack(虚拟机栈JVM Stack):

  • 每个线程会对应一个Java栈;
  • 每个Java栈由若干栈帧组成;
  • 每个方法对应一个栈帧;
  • 栈帧在方法运行时,创建并入栈;方法执行完,该栈帧弹出栈帧中的元素作为该方法返回值,该栈帧被清除;
  • 栈顶的栈帧叫活动栈,表示当前执行的方法,才可以被CPU执行;
  • 线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常;
  • 栈扩展时无法申请到足够的内存,就会抛出OutOfMemoryError异常;

 

2.3.方法区MethodArea

  • 方法区是Java堆的永久区(PermanetGeneration)
  • 方法区存放了要加载的类的信息(名称、修饰符等)、类中的静态常量、类中定义为final类型的常量、类中的Field信息、类中的方法信息,
  • 方法区是被Java线程共享的
  • 方法区要使用的内存超过其允许的大小时,会抛出OutOfMemoryError: PremGen space的错误信息。

 

2.4.常量池ConstantPool:

 

        字面量:字符串、final变量等。

        引用量:类/接口、方法和字段的名称和描述符,

  • 常量池是方法区的一部分。
  • 常量池中存储两类数据:字面量和引用量。
  • 常量池在编译期间就被确定,并保存在已编译的.class文件中

 

2.5.本地方法栈Native Method Stack:

  1. 本地方法栈和Java栈所发挥的作用非常相似,区别不过是Java栈为JVM执行Java方法服务,而本地方法栈为JVM执行Native方法服务。
  2. 本地方法栈也会抛出StackOverflowError和OutOfMemoryError异常。

 

3.Java内存模型工作示意图

 

 

 

  1. 首先类加载器将Java代码加载到方法区
  2. 然后执行引擎从方法区找到main方法
  3. 为方法创建栈帧放入方法栈,同时创建该栈帧的程序计数器
  4. 执行引擎请求CPU执行该方法
  5. CPU将方法栈数据加载到工作内存(寄存器和高速缓存),执行该方法
  6. CPU执行完之后将执行结果从工作内存同步到主内存

 

 

 

 

 

 

 

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值