面试官:JDK8内存分布

内存组成部分

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

image.png

虚拟机栈

每个线程都有一私有栈,随着线程的创建而创建。栈里面会存放着“栈帧”,每个方法创建一个栈帧。
栈帧的组成部分:
一、局部变量:存放基本的数据类型(bytecharintdouble,float,boolean,long,short)和对象地址引用(一个指针指向对象或者一条指令)。局部变量区被定义一个从0开始的数字数组,byteCharshortboolean转换成intlong ,double, 2个字节。其中64 位长度的longdouble 类型的数据会占用2个局部变量空间(Slot),其余的数据类型只占用1个。局部变量通过数组的下标访问。
备注:局部变量表中的Slot是可重用的,方法体中定义的变量,其作用域并不一定会覆盖整个方法,如果当前字节码PC计算器的值已经超出了某个变量的作用域,那么这个变量对应的Slot就可以交给其它变量使用。
二、操作数栈:数组,先进后出,pushpop操作。
三、帧数据区:
1、解析常量池的数据
2、方法执行完后处理方法返回,恢复调用方现场
3、方法执行过程中抛出异常时异常的处理,当出现异常时虚拟机查找相应的异常表看是否有对应的catch语句,如果没有就抛出异常终止这个方法调用。
如果栈内存不够将会抛出StackOverflowError错误。catch 捕获的是 Throwable类。
调整栈内存大小参数-Xss:每个线程的stack大小(栈)

虚拟机堆

堆内存是 JVM所有线程共享的部分,在虚拟机启动的时候就已经创建。所有的对象和数组都在堆上进行分配。这部分空间可通过 GC 进行回收。
当申请不到空间时会抛出 OutOfMemoryErrorcatch捕获的是 Throwable
调整堆参数大小参数:
XmxJava Heap的最大值、默认为物理内存的1/4
XmsJava Heap的初始值,server端最好Xms与Xmx一样
XmnJava Heapyoung区的大小
XX:MetaspaceSize 元数据初始大小
XX:MaxMetaspaceSize 元数据内存最大值
JDK1.8中,取消了PermGen,取而代之的是Metaspace
元数据空间并不在虚拟机中,而是使用本地内存。

方法区

方法区也是所有线程共享。主要用于存储类的信息、常量池、方法数据、方法代码等。方法区逻辑上属于堆的一部分,但是为了与堆进行区分,通常又叫“非堆”。

本地方法区

这部分主要与虚拟机用到的 Native方法相关,一般情况下, Java 应用程序员并不需要关心这部分的内容。

程序计数器

JVM支持多个线程同时运行,每个线程都有自己的程序计数器。倘若当前执行的是 JVM 的方法,则该寄存器中保存当前执行指令的地址;倘若执行的是Native方法,则程序计数器中为空。

举例

image.png

你可能感兴趣的:(面试官:JDK8内存分布)