jvm内存模型及参数

内存模型

JVM

堆,栈,方法区,计数器

  • 线程共享

存放所有对象实例,GC的主要区域,可以分为新生代,老年代,新生代可以分为一个eden,两个survivor空间(from survivor,to survivor),不需要连续内存,使用-XMX和-XMS控制空间最大最小,推荐写一样大,减少堆空间切换过程的消耗

  • 栈(本地方法栈,虚拟机栈) 线程私有

存放局部变量,操作数栈,动态链接,方法出口等信息

  • 方法区(运行时常量池)线程共享

存放类的基本信息,静态变量,常量,即时编译器编译后的代码等

  • 计数器 线程私有

记录正在执行的虚拟机字节码指令的地址,native方法为空

  • GC线程共享

新生代使用复制算法
老年代使用标记-清理或标记-整理算法

  • 内存分配:
  1. 大多数情况,对象直接在新生代的Eden空间分配,当eden空间不够时,发起Minor GC,因为新生代对象存活时间很短所以会频繁发生Minor GC,其执行速度也会比较快。
  2. 大对象会直接分配到老年代中,大对象指需要连续内存空间的对象,如很长的字符串或数组,可以通过指令-XX:PretenureSizeThreshold参数控制大小分界线,老年代会执行Full GC
  3. 经过Minor GC依然存活,且能被Survivor区容纳的对象,会被转移到Survivor区,标记计数器就增加1,默认增加到15时会转移到老年代,可以通过-XX:MaxTenuringThreshold设置。
  4. JVM在执行Minor GC之前,会检查老年代剩余空间>新生代总空间,如果大于那么Minor GC是安全的,如果小于则会查看指令HandlePromotionFailure是否允许担保失败,如果允许则继续检查老年代剩余空间>历次晋升到老年代对象的平均大小,如果大于,则尝试进行Minor GC,如果小于或不允许,则进行一次FULL GC,使老年代清除一次垃圾
  5. 调用System.gc()也会触发Full GC,Full GC后空间仍然不足,则会抛出OutOfMemoryError,调优时,尽量让对象在Minor GC期间被回收,让对象在新生代多存活,不要使用大对象

你可能感兴趣的:(jvm内存模型及参数)