JVM运行时数据区(RuntimeDataArea)
这就是我们常说的JVM的内存了,它主要分为五个部分
1、堆(Heap):一个java虚拟机实例中只存在一个堆内存空间,被所有线程共享;堆是运行时数据区域,所有类的实例和数组的内存均从此处分配,堆是在java虚拟机启动时创建的,在堆内存之外的内存成为非堆内存。对象的堆内存右垃圾回收器自动管理系统回收,堆的大小可以固定,也可以扩大和缩小,堆的内存不需要是连续空间。
堆内存分配:
- JVM初始分配的堆内存有-Xms指定,默认是物理内存的1/64;
- JVM最大分配的堆内存由-Xmx指定,默认是物理内存的1/4;
- 默认空余堆内存小于40%时,JVM就会增大堆直到-Xmx指定的最大限制;
- 空余堆内存大于70%时,JVM会减少堆直到-Xms指定的最小限制;
- 服务器一般可以设置-Xms、-Xmx相等以避免在每次GC后调整堆的大小;
- 备注:如果-Xmx不指定或者指定偏小,引用可能会导致java.lang.OutOfMemory错误,此错误来自JVM,不是Throwable的,无法用try...catch捕捉。
内存设置参数:
• -Xms:例:-Xms512m,表示JVM初始分配的堆内存大小为512m(JVM堆内存最小尺寸,初始分配);
• -Xmx:例:-Xmx1024m,表示JVM最大允许分配的堆内存大小为1024m,按需分配,默认我物理内存的1/4;
• -XX:PermSize:例:-XX:PermSize=512M,JVM初始分配的非堆内存;
• -XX:MaxPermSize:例:-XX:MaxPermSize=1024M,表示JVM最大允许分配的非堆内存,按需分配;
• -XX:NewSize:表示JVM启动时YOUNG(年轻代)的内存大小;
• -XX:MaxNewSize:表示最大可占用的YOUNG(年轻代)内存大小;
• -XX:SurvivorRatio:设置YOUNG代中Survivor空间和Eden空间的比例
2、方法区域(MethodArea):方法区与堆一样,也是被线程共享的区域,被装载的class的信息存储在方法去内存中,存储了每个类的信息,这些信息包括类的名称、方法信息、字段信息、静态变量、常量、类型信息以及编译器编译后的代码等。当虚拟机装载某个类型时,它使用类装载器定位相应的class文件,然后读入这个class文件内容并把它传输到虚拟机中。
3、栈(JavaStack):每当启用一个线程时,JVM就为他分配一个Java栈,栈是以帧为单位保存当前线程的运行状态。某个线程正在执行的方法称为当前方法,当前方法使用的栈帧称为当前帧,当前方法所属的类称为当前类,当前类的常量池称为当前常量池。当线程执行一个方法时,它会跟踪当前常量池。
每当线程调用一个Java方法时,JVM就会在该线程对应的栈中压入一个帧,这个帧自然就成了当前帧。当执行这个方法时,它使用这个帧来存储参数、局部变量、中间运算结果等等。 Java栈上的所有数据都是私有的。任何线程都不能访问另一个线程的栈数据。所以我们不用考虑多线程情况下栈数据访问同步的情况。虚拟机只会直接对栈执行两种操作,以帧为单位的压栈或出栈
4、程序计数器(ProgramCounter):每一个线程都有它自己的PC寄存器,也是该线程启动时创建的。PC寄存器的内容总是指向下一条将被执行指令的饿地址,这里的地址可以是一个本地指针,也可以是在方法区中相对应于该方法起始指令的偏移量。
5、本地方法栈(Nativemethodstack):保存native方法进入区域的地址。
最后附上一张JVM内存模型图