java

关注公众号 jb51net

关闭
首页 > 软件编程 > java > JVM堆区

简单说说JVM堆区的相关知识

作者:ayugudu

今天给大家带来的是关于Java虚拟机的相关知识,文章围绕着JVM堆区展开,文中有非常详细的介绍及代码示例,需要的朋友可以参考下

一、堆概述

1.1 堆内存细分

现代垃圾收集器大部分基于分代收集理论设计,堆空间细分为:

使用下面命令设置堆空间初始化 10m,最大空间 10m

-Xms10m -Xmx10m

使用java visual 查看 visual gc

可以看出通过参数设置的内存大小 只与新生代(Eden+s0+s1 ),老年代有关,而在逻辑上还要加上元空间。

1.2 堆空间大小的设置

1.2.1 通过参数设置

-Xms 用来设置堆空间(年轻代+老年代)的初始内存大小
  -X 是jvm的运行参数
  ms 是memory start
  
-Xmx 用来设置堆空间(年轻代+老年代)的初始内存大小

1.2.2 默认空间大小

使用一下代码查看 当前jvm初始化内存与最大内存

/**
 * @program: jvmDemo
 * @description:
 * @author: wfg
 * @create: 2021-06-14 10:40
 */
public class Test9 {

    public static void main(String[] args) {
        //返回jvm中的内存总量(字节)
          long initialMemory = Runtime.getRuntime().totalMemory()/1024/1024;
        //虚拟机将尝试使用最大堆内存
         long maxMemory = Runtime.getRuntime().maxMemory()/1024/1024;
        System.out.println("-Xms:"+initialMemory+"m");

        System.out.println("-Xmx:"+maxMemory+"m");


        System.out.println("系统大小:"+initialMemory*64/1024+"G");
        System.out.println("系统大小:"+maxMemory*4/1024+"G");

    }

}

结果(本机运行内存为 8g)

1.2.3 通过参数设置堆空间大小后内存不一致问题

设置300

查看

/**
 * @program: jvmDemo
 * @description:
 * @author: wfg
 * @create: 2021-06-14 10:40
 */
public class Test9 {

    public static void main(String[] args) {
        //返回jvm中的内存总量(字节)
        long initialMemory = Runtime.getRuntime().totalMemory() / 1024 / 1024;
        //虚拟机将尝试使用最大堆内存
        long maxMemory = Runtime.getRuntime().maxMemory() / 1024 / 1024;
        
        System.out.println("-Xms:" + initialMemory + "m");

        System.out.println("-Xmx:" + maxMemory + "m");
        
    }

}

结果

分析

在vm参数设置里面加上

-XX:+PrintGCDetails 

再次运行程序查看

原理

新生代的s0 和 s1 只能有一个生效

1.3 年轻代与年老代

一类是生命周期较短的瞬时对象。

另外一类是对象生命周期非常长。

可以通过 -XX:NewRation设置新生代与老年代的比例,默认值是2.(一般都不会去设置)

1.4 对象分配过程

1.new的对象先放伊甸区,此区有大小限制

2.当伊甸区满的时候,程序需要创建时,jvm的垃圾回收将对伊甸园区进行垃圾回收(minor gc)

3.然后将伊甸园中的剩余对象移动到辛存者0

4.如果再次触发垃圾回收,上次幸存下来的放到幸存者0区,没有回收,就会放到幸存者1区

5.再次经历垃圾回收会重新放回辛存者0区

6.当在辛存者区达到15次时,就可以去老年区了

可以设置参数:-XX:MaxTenuringThreshold=

7.当养老区内存不足时,再次触发GC:major GC,进行养老区的内存处理

8.若养老区进行处理后,依然无法进行对象的保存,就会产生00m异常

java.lang.outofMemoryError:java heap space

9.总结

10.流程

当Eden满时,会触发MinorGC算法来回收memory,旨在清理掉再无引用的数据(在内存里是Tree),意图存储到S0. 若此时S0也满了,会再次MinorGC意图回收S0无引用的数据,把有引用的数据移动到S1。如果S1够用,此时会清空S0;如果S1满了,会回滚刚存入S1的数据,直接把本次GC的数据存入Old区,S0保持刚刚MinorGC时的状态。延伸:如果Old也满了,会触发MajorGC,如果还是不够,则存入Permanent Generate,不幸这里也满了,会在允许的范围内按照内置的规则自动增长,可能不会发生GC,也可能会。当增长的量不够存时,会触发Full GC。若FullGC后还是不够存,自动增长的量也超过了允许的范围,则发生内存溢出。还有一种情况,就是分配的线程栈处于很深的递归或死循环时,会发生栈内存溢出。

二、对象分配过程:Tlab

2.1 为什么要有tlab

2.2什么是tlab

对象分配流程

对象分配流程

三、堆空间常用参数设置

四、堆是分配对象的唯一选择吗

  1. 随着jit编译期的发展与逃逸分析技术成熟,栈上分配与标量替换优化技术导致所有对象都分配到堆上不那么绝对了
  2. 如果经过逃逸分析后发现,一个对象并没有逃逸出方法的话,那么可能被优化成栈上分配

4.1 逃逸分析

判断逃逸的方法:看new 的对象实体是否有可能在方法外被调用

在这里插入图片描述

4.2 代码优化

到此这篇关于简单说说JVM堆区的相关知识的文章就介绍到这了,更多相关JVM堆区内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文