浅谈JVM(一)——JVM内存模型

6 篇文章 0 订阅

JVM内存模型如下图所示:

这里写图片描述

JVM运行时,内存可以分为如下几个部分:

  • 堆(Heap)
  • 方法区(Method Area)
  • 虚拟机栈(Java Stack)
  • 本地方法区(Native Method Stack)
  • 程序计数器(PC Register)

堆(Heap)和方法区

堆(Heap)

Java堆是绝大多数Java开发人员最关注的。堆是属于线程共享的一块内存区域,它在JVM启动的时候被创建,并且在实际空间中是不连续的。

作用:用来存储对象实例的内存区,也是GC执行垃圾回收的重点区域。而堆内存又可以分为新生代、老年代和永久代(也叫方法区),其中新生代又可以分为Eden,From SurvivorTo Survivor,如下图所示:

这里写图片描述

Eden:用于存放声明周期较短的对象。
Survivor:垃圾对象回收的交换空间。
老年代:存放周期比较长或者比较大的对象。

注意:大家可以通过“-Xms”表示堆区的起始内存,“-Xmx”表示堆区最大内存。如果内存超过指定的最大内存,就会抛出OutOfMemoryError异常。

方法区

方法区和堆一样,也是允许被所有的线程共享,在虚拟机HotSpot上,他俩仅仅是逻辑上的独立,实际上方法区是包含在堆区的。

作用:存储Java类对象信息,比如:运行常量池、字段、和方法数据、构造函数和普通方法字节码和一些特殊方法。

方法区也是在JVM创建的时候被创建,方法区也被叫做永久代,这是因为方法区很少被GC执行,并且默认情况下GC回收的目标仅仅是方法区中的常量池和类型卸载。

注意:通过“-XX:MaxPermSize”设置内存的大小,一旦超出,则会抛出OutOfMemoryError。


程序计数器(PC Register)

程序计数器是属于线程私有内存区的,生命周期与线程的生命周期保持一致。如果当前线程正在执行JAVA方法,则PC寄存器就会记录此时执行的字节码指令地址。如果是native方法,则为空。

作用:在多线程环境下,准确记录各个线程执行的当前字节码指令地址,使得CPU可以不停的切换不同线程执行任务而不会出错。

注意:不会出现OutOfMemoryError


虚拟机栈(Java Stack)、本地方法区(Native Method Stack)

Java栈也可以称为Java虚拟机栈,和PC寄存器一样,也是线程私有的,并且生命周期和线程保持一致。

作用:用于存储栈帧(Stack Frame),而栈帧所存储的就是局部变量表、操作数栈,和方法出口。而局部变量表用于存储各类原始数据类型、对象引用等。

和Java栈一样,本地方法区也是线程私有的,用于支持本地方法的执行,作用和Java栈类似。如果JVM不打算支持native方法,也不依赖传统栈,就可以不实现本地方法栈。

注意:Java栈和本地方法栈一样,内存允许固定或者动态扩展。一旦内存固定,如果线程请求的栈内存过大,则会抛出StackOverFlowError
,反之抛出OutOfMemoryError异常。


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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值