[JVM内存结构模型][调参优化][垃圾回收机制]

1. JVM内存结构模型图

在这里插入图片描述

2. 栈stack

  • 每个线程都会为自己创建一个私有的JVM栈,当前线程中每一次的方法调用都会创建出一个新的栈帧,每个栈帧中都会有自己的局部变量,操作数栈,动态链接,方法出口等等,当方法结束后当前栈帧也就会被销毁掉。
  • 局部变量:每一个栈帧都会包含一个本地变量的数组,用于储存本地变量,本地变量的大小是在编译的时候确定的。
  • 操作数栈:每一个栈帧都会包含一个操作数栈,jvm会将变量的值加载到操作数栈上,对变量值进行操作,然后对应的结果就会出栈赋值给我们的变量。
  • 动态链接:class文件中方法的调用和变量的访问,是用符号引用实现的,动态链接就是把这些符号方法引用转为具体的引用。
  • 方法出口:当前帧执行完成后会执行return指令,返回到调用帧执行的位置。
public class Stack {
    static int i = 0;

    public static void main(String[] args) throws Throwable{
        Stack stack = new Stack();

        try {
            stack.doStack();
        } finally {
            System.out.println(i);
        }
    }

    public void doStack() {
        i++;
        doStack();
    }
}
设置栈的大小为125k
console:
990
Exception in thread "main" java.lang.StackOverflowError
设置栈的大小为256k
console:
4097
Exception in thread "main" java.lang.StackOverflowError
  • 栈大小设置过小那么能分配的栈帧的也会变少,容易出现StackOverflowError,但是能创建的线程就会相对较多。
  • 栈大小设置过大那么能创建的线程就会相对较少,容易出现内存溢出。

3. 堆heap

  1. 一般来说new出来的对象都会存储堆中,对应的地址值会储存在栈中。
  2. Thread Local Allocation Buffer技术,每个线程在Java堆中预先分配一小块内存TLAB,是线程的独立空间,用于保证多线程下内存分配安全。
    在这里插入图片描述

4. 元空间/方法区

  • 元空间用来存储常量,静态变量,类元信息。储存在本地内存中默认的初始大小是21M,所以元空间的大小受到本地内存大小的限制,元空间满后会触发Full GC。

5. 垃圾回收机制GC

  • Eden内存满厚会触发Minor GC,每一次的Minor GC后存活对象都会向又偏移到Survivor0,Survivor0内存满后再次触发Minor GC存活的对象移动到Survivor1。年轻代的对象会逐渐复制到老年代中,当老年代的内存空间不足时会触发Full GC。如果Full Gc后,老年代的内存还是不足就会出现内存溢出OOM内存溢出。Minor GC时间校对,Full GC时间较长。
  • 对象的回收,大多数的对象还是存放在我们的堆中的,在做回收之前需要盘判断这个对象是否还有其他的应用,若无引用则可以被回收。例如我们刚 Student student = new Student(); 然后student = null;那么我们的这种对象就是无引用的对象了,是会被回收掉的。
  • 引用计数器:引用到就会+1,引用失效后-1,当值为0时则表示为垃圾对象,GC会回收。但是会因为相互嵌套依赖而导致无法被回收。
  • 可达性分析:可根据GC Roots节点能查询到的对象都不是垃圾对象,当一个对象和GC Roots不可达时,则该对象是无用的。

6. 参数优化

  • 为了避免OOM和StackOverflowError,或者是频繁触发Full GC我们需要合理的对JVM分配内存空间。
  1. 设置堆内存的初始化大小(-Xms 单位M、G)
  2. 设置堆内存最大内存大小(-MaxHeapSize -Xmx 建议不要超过物理内存的80%)
  3. 设置年轻代最大内存大小(-XX:MaxNewSize -Xmn)
  4. 设置栈内存大小(-Xss)避免栈内存过小触发StackOverflowError
  5. 设置元空间初始内存大小(-XX:MetaspaceSize)元空间使用的是本地内存。
  6. 设置元空间最大内存大小(-XX:MaxMetaspaceSize)
  • 根据堆内存大小设置年轻代的初始化大小(-XX:NewSize -Xns)和年轻代最大内存大小(-XX:MaxNewSize -Mnx)
    建议参数: ‐Xms3072M ‐Xmx3072M ‐Xmn2048M ‐Xss1M ‐XX:MetaspaceSize=256M ‐XX:MaxMetaspaceSize=256M
  • 我们应该让对象在年轻代的时候就应该被GC回收,不要让对象因为频繁的GC进入到老年代,因为每次的Full GC都会暂停其他线程(Stop the World)导致系统全局卡顿。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值