JVM垃圾回收相关及堆分代原理

简介: JVM垃圾回收相关及堆分代原理

上次咱们提到了垃圾回收(将内存中不再被使用的对象进行回收),GC(Garbage Collection)分为两种,一种是很快的对新生代对象收集的叫minor GC,另一种是很慢的对旧生代对象收集叫Full GC,触发Full GC可以使用System.gc()


是怎么收集不用的内存呢?垃圾收集算法,第一种最低效的叫,标记清除算法,首先是他标记清除的效率都不高,另外,看图上,清除后会留下不连续空白,如果不是最后几个内容空间被清除,一旦有大对象需要连续的内存空间存的时候,会发现没地方存的下,导致提前触发下一次gc。

有了这个问题,咱们可以接着看下一个算法,复制算法,步骤是这样的,首先把内存划分为两块完全相同的部分,前面和上面的标记清除算法一致,但是回收之后做了一个操作,将上面部分清除后存活的对象全部复制到了下面,达到没有连续空白空间的作用。问题也很明显,可以使用的内存变成了原来的一半,另外还有一个多了一个复制的时间。

我们继续带着问题往下看下面这个算法,标记整理算法,步骤是这样的,先标记需要清除的内存,让存活的对象往一端移动,清除掉存活对象之外的空间,让前面内存不留空白。

有了垃圾收集算法的基础,我们可以开始了解下,堆内存的分代是怎么划分的了,下面这个图可以很清楚的看到堆内存各代的划分,其中持久代(永久存储区)主要存放的是Java类的类信息,与垃圾收集要收集的Java对象关系不大。年轻代(新生区)和年老代(养老区)的划分是对垃圾收集影响比较大的。




新生代:每次GC时都会有大量对象死去,少量存活,使用复制算法。新生代又分为Eden区和Survivor区(Survivor from、Survivor to),大小比例默认为8:1:1。

:在年轻代中经历了N次垃圾回收后仍然存活的对象,就会被放到年老代中。因此,可以认为年老代中存放的都是一些生命周期较长的对象。对象存活率高、没有额外空间进行分配担保,就使用标记-清除或标记-整理算法。

持久:用于存放静态文件,如今Java类、方法等。持久代对垃圾回收没有显著影响,但是有些应用可能动态生成或者调用一些class,例如Hibernate等,在这种时候需要设置一个比较大的持久代空间来存放这些运行过程中新增的类。持久代大小通过-XX:MaxPermSize=<N>进行设置。

接着咱们来看看他们新生代和年老代间是怎么转换,

我们直接新new的对象最开始是在Eden区的,和他的名字一样,伊甸园,一起梦开始的地方哈哈哈,Eden满了咋办?这时候接着用其中一个Survivor from区,也叫存活区,如果Survivor from也满了怎么办?这时候就触发新生代的GC(Minor GC前面提到过是新生代的时候触发的),这时候会有一堆对象死掉,被释放掉,还会有存活的对象,将Eden和Survivor from中存活的对象一起复制到另一个Survivor to中,然后清空EdenSurvivor from区,接着重新开始给Eden区存新对象,Eden满了在给Survivor to存。发现了没,精妙吧,竟然形成了一个循环,每次都有一个Survivor 是空的,Eden和另一个Survivor 在存,满了触发GC复制到空的Survivor 里无限循环,太精秒了。


新生代到老年代的转换方法:

1.那么问题来了,如果,空的那个Survivor 容纳不了刚才Eden和另一个不空的Survivor GC后存活的对象怎么办?别担心,年代再给你兜底呢,直接将这一堆放不下的对象放老年代里,如果老年代也放不下了,触发Full GC(老年代GC)。


2.新生代给老年代转换,不仅可以上面存不下在转,还有几种方式,JVM其实给每个对象都有一个年龄(逃过GC的次数)的计数器,从Eden出生,第一次Minor GC没把对象干掉,移到Survivor区后年龄就加1,以后每次逃过一次Minor GC,年龄就加1,默认是15岁(通过XX:MaxTenuringThreshold来设定),到这年龄就算资历很深了,可以算老人了,进入到老年代。


3.还有个进入老年代的机会,如果Survivor 空间中相同年龄所有对象的大小的总和大于Survivor的大小一半,年龄大于等于x的所有对象直接进入老年代,无需等到最大年龄要求。


4.另外,很大的对象是可以直接进入老年代的,在JVM中有个参数配置-XX:PretenureSizeThreshold,只要对象大小大于这个设置的值,直接进入老年代,避免在Eden和Survivor区之间发生大量的内存复制,咋样精妙吧,这设计可以吧哈哈哈。


最后,大家想获取更多知识的,可以继续关注公众号,不定时推送。分享了这么牛逼的知识,还不请小编喝个水吗,哈哈哈,欢迎土豪直接赏赞,谢谢,您的支持就是小编最大的动力。

相关文章
|
10月前
|
Arthas 存储 算法
深入理解JVM,包含字节码文件,内存结构,垃圾回收,类的声明周期,类加载器
JVM全称是Java Virtual Machine-Java虚拟机JVM作用:本质上是一个运行在计算机上的程序,职责是运行Java字节码文件,编译为机器码交由计算机运行类的生命周期概述:类的生命周期描述了一个类加载,使用,卸载的整个过类的生命周期阶段:类的声明周期主要分为五个阶段:加载->连接->初始化->使用->卸载,其中连接中分为三个小阶段验证->准备->解析类加载器的定义:JVM提供类加载器给Java程序去获取类和接口字节码数据类加载器的作用:类加载器接受字节码文件。
868 55
|
10月前
|
Oracle Java 关系型数据库
JVM深入原理(一+二):JVM概述和JVM功能
JVM全称是Java Virtual Machine-Java虚拟机JVM作用:本质上是一个运行在计算机上的程序,职责是运行Java字节码文件,编译为机器码交由计算机运行。
263 0
|
10月前
|
Arthas 存储 Java
JVM深入原理(三+四):JVM组成和JVM字节码文件
目录3. JVM组成3.1. 组成-运行时数据区3.2. 组成-类加载器3.3. 组成-执行引擎3.4. 组成-本地接口4. JVM字节码文件4.1. 字节码文件-组成4.1.1. 组成-基础信息4.1.1.1. 基础信息-魔数4.1.1.2. 基础信息-主副版本号4.1.2. 组成-常量池4.1.3. 组成-方法4.1.3.1. 方法-工作流程4.1.4. 组成-字段4.1.5. 组成-属性4.2. 字节码文件-查看工具4.2.1. javap4.2.2. jclasslib4.2.3. 阿里Arthas
179 0
|
10月前
|
存储 安全 Java
JVM深入原理(五):JVM组成和JVM字节码文件
类的生命周期概述:类的生命周期描述了一个类加载,使用,卸载的整个过类的生命周期阶段:类的声明周期主要分为五个阶段:加载->连接->初始化->使用->卸载,其中连接中分为三个小阶段验证->准备->解析。
148 0
|
10月前
|
Arthas Java 测试技术
JVM深入原理(六)(一):JVM类加载器
目录6. JVM类加载器6.1. 类加载器-概述6.2. 类加载器-执行流程6.3. 类加载器-分类(JDK8)6.3.1. JVM底层实现的类加载器6.3.1.1. 启动类加载器6.3.2. Java代码实现类的加载器6.3.2.1. 扩展类加载器6.3.2.2. 应用程序类加载器6.4. 类加载器-Arthas查看类加载器
186 0
|
10月前
|
Java 关系型数据库 MySQL
JVM深入原理(六)(二):双亲委派机制
自定义类加载器打破双亲委派机制的方法:复写ClassLoader中的loadClass方法常见问题:要加载的类名如果是以java.开头,则会抛出安全性异常加载自定义的类都会有一个共同的父类Object,需要在代码中交由父类加载器去加载自定义类加载器不手动指定parent会默认指定应用类加载两个自定义类加载器加载同一个类会被认为是两个对象,只有相同的类加载器+想通的类限定名才会被认为是一个对象。
334 0
|
10月前
|
存储 安全 Java
JVM深入原理(七)(一):运行时数据区
栈的介绍:Java虚拟机栈采用栈的数据结构来管理方法调用中的基本数据,先进后出,每一个方法的调用使用一个栈帧来保存栈的组成:栈:一个线程运行所需要的内存空间,一个栈由多个栈帧组成栈帧:一个方法运行所需要的内存空间活动栈帧:一个线程中只能有一个活动栈帧栈的生命周期:栈随着线程的创建而创建,而回收会在线程销毁时进行栈的执行流程:栈帧压入栈内执行方法执行完毕释放内存若方法间存在调用,那么会压入被调用方法入栈,执行完后释放内存,再执行当前方法,直到执行完毕,释放所有内存。
195 0
|
10月前
|
存储 缓存 安全
JVM深入原理(七)(二):运行时数据区
堆的作用:存放对象的内存空间,它是空间最大的一块内存区域.栈上的局部变量表中,可以存放堆上对象的引用。静态变量也可以存放堆对象的引用,通过静态变量就可以实现对象在线程之间共享。堆的特点:线程共享:堆中的对象都需要考虑线程安全的问题垃圾回收:堆有垃圾回收机制,不再引用的对象就会被回收方法区的概述:方法区是存放基础信息的位置,线程共享,主要包括:类的元信息:保存了所有类的基本信息运行时常量池:保存了字节码文件中的常量池内容静态常量池:字节码文件通过编号查表的方式找到常量。
139 0
|
10月前
|
缓存 算法 Java
JVM深入原理(八)(一):垃圾回收
弱引用-作用:JVM中使用WeakReference对象来实现软引用,一般在ThreadLocal中,当进行垃圾回收时,被弱引用对象引用的对象就直接被回收.软引用-作用:JVM中使用SoftReference对象来实现软引用,一般在缓存中使用,当程序内存不足时,被引用的对象就会被回收.强引用-作用:可达性算法描述的根对象引用普通对象的引用,指的就是强引用,只要有这层关系存在,被引用的对象就会不被垃圾回收。引用计数法-缺点:如果两个对象循环引用,而又没有其他的对象来引用它们,这样就造成垃圾堆积。
242 0
|
10月前
|
算法 Java 对象存储
JVM深入原理(八)(二):垃圾回收
Java垃圾回收过程会通过单独的GC线程来完成,但是不管使用哪一种GC算法,都会有部分阶段需要停止所有的用户线程。这个过程被称之为StopTheWorld简称STW,如果STW时间过长则会影响用户的使用。一般来说,堆内存越大,最大STW就越长,想减少最大STW,就会减少吞吐量,不同的GC算法适用于不同的场景。分代回收算法将整个堆中的区域划分为新生代和老年代。--超过新生代大小的大对象会直接晋升到老年代。
239 0