内存组成部分
根据 JVM
规范,JVM
内存共分为虚拟机栈、堆、方法区、程序计数器、本地方法栈五个部分。
虚拟机栈
每个线程都有一私有栈,随着线程的创建而创建。栈里面会存放着“栈帧”,每个方法创建一个栈帧。
栈帧的组成部分:
一、局部变量:存放基本的数据类型(byte
,char
,int
,double
,float
,boolean
,long
,short
)和对象地址引用(一个指针指向对象或者一条指令)。局部变量区被定义一个从0开始的数字数组,byte
、Char
、short
,boolean
转换成int
,long
,double
, 2
个字节。其中64
位长度的long
和double
类型的数据会占用2
个局部变量空间(Slot
),其余的数据类型只占用1个。局部变量通过数组的下标访问。
备注:局部变量表中的Slot
是可重用的,方法体中定义的变量,其作用域并不一定会覆盖整个方法,如果当前字节码PC计算器的值已经超出了某个变量的作用域,那么这个变量对应的Slot
就可以交给其它变量使用。
二、操作数栈:数组,先进后出,push
和pop
操作。
三、帧数据区:
1、解析常量池的数据
2、方法执行完后处理方法返回,恢复调用方现场
3、方法执行过程中抛出异常时异常的处理,当出现异常时虚拟机查找相应的异常表看是否有对应的catch
语句,如果没有就抛出异常终止这个方法调用。
如果栈内存不够将会抛出StackOverflowError
错误。catch
捕获的是 Throwable
类。
调整栈内存大小参数-Xss
:每个线程的stack大小(栈)
虚拟机堆
堆内存是 JVM
所有线程共享的部分,在虚拟机启动的时候就已经创建。所有的对象和数组都在堆上进行分配。这部分空间可通过 GC
进行回收。
当申请不到空间时会抛出 OutOfMemoryError
。catch
捕获的是 Throwable
。
调整堆参数大小参数:
Xmx
:Java Heap
的最大值、默认为物理内存的1/4
Xms
:Java Heap
的初始值,server端最好Xms与Xmx一样
Xmn
:Java Heap
young区的大小
XX:MetaspaceSize
元数据初始大小
XX:MaxMetaspaceSize
元数据内存最大值
在JDK1.8
中,取消了PermGen
,取而代之的是Metaspace
。
元数据空间并不在虚拟机中,而是使用本地内存。
方法区
方法区也是所有线程共享。主要用于存储类的信息、常量池、方法数据、方法代码等。方法区逻辑上属于堆的一部分,但是为了与堆进行区分,通常又叫“非堆”。
本地方法区
这部分主要与虚拟机用到的 Native
方法相关,一般情况下, Java
应用程序员并不需要关心这部分的内容。
程序计数器
JVM
支持多个线程同时运行,每个线程都有自己的程序计数器。倘若当前执行的是 JVM
的方法,则该寄存器中保存当前执行指令的地址;倘若执行的是Native
方法,则程序计数器中为空。